diff --git a/.project b/.project
new file mode 100644
index 0000000000..9a31037252
--- /dev/null
+++ b/.project
@@ -0,0 +1,11 @@
+
+
+ esp-idf(1)
+
+
+
+
+
+
+
+
diff --git a/IDF_README_Backup.md b/IDF_README_Backup.md
new file mode 100644
index 0000000000..42d8f45b4d
--- /dev/null
+++ b/IDF_README_Backup.md
@@ -0,0 +1,124 @@
+# Espressif IoT Development Framework
+
+[](https://docs.espressif.com/projects/esp-idf/en/latest/?badge=latest)
+
+ESP-IDF is the official development framework for the [ESP32](https://espressif.com/en/products/hardware/esp32/overview) chip.
+
+# Developing With ESP-IDF
+
+## Setting Up ESP-IDF
+
+See setup guides for detailed instructions to set up the ESP-IDF:
+
+* [Getting Started Guide for the stable ESP-IDF version](https://docs.espressif.com/projects/esp-idf/en/stable/get-started/)
+* [Getting Started Guide for the latest (master branch) ESP-IDF version](https://docs.espressif.com/projects/esp-idf/en/latest/get-started/)
+
+## Finding a Project
+
+As well as the [esp-idf-template](https://github.com/espressif/esp-idf-template) project mentioned in Getting Started, ESP-IDF comes with some example projects in the [examples](examples) directory.
+
+Once you've found the project you want to work with, change to its directory and you can configure and build it.
+
+To start your own project based on an example, copy the example project directory outside of the ESP-IDF directory.
+
+# Quick Reference
+
+See the Getting Started guide links above for a detailed setup guide. This is a quick reference for common commands when working with ESP-IDF projects:
+
+## Configuring the Project
+
+`make menuconfig`
+
+* Opens a text-based configuration menu for the project.
+* Use up & down arrow keys to navigate the menu.
+* Use Enter key to go into a submenu, Escape key to go out or to exit.
+* Type `?` to see a help screen. Enter key exits the help screen.
+* Use Space key, or `Y` and `N` keys to enable (Yes) and disable (No) configuration items with checkboxes "`[*]`"
+* Pressing `?` while highlighting a configuration item displays help about that item.
+* Type `/` to search the configuration items.
+
+Once done configuring, press Escape multiple times to exit and say "Yes" to save the new configuration when prompted.
+
+## Compiling the Project
+
+`make -j4 all`
+
+... will compile app, bootloader and generate a partition table based on the config.
+
+NOTE: The `-j4` option causes `make` to run 4 parallel jobs. This is much faster than the default single job. The recommended number to pass to this option is `-j(number of CPUs + 1)`.
+
+## Flashing the Project
+
+When `make all` finishes, it will print a command line to use esptool.py to flash the chip. However you can also do this from make by running:
+
+`make -j4 flash`
+
+This will flash the entire project (app, bootloader and partition table) to a new chip. The settings for serial port flashing can be configured with `make menuconfig`.
+
+You don't need to run `make all` before running `make flash`, `make flash` will automatically rebuild anything which needs it.
+
+## Viewing Serial Output
+
+The `make monitor` target uses the [idf_monitor tool](https://docs.espressif.com/projects/esp-idf/en/latest/get-started/idf-monitor.html) to display serial output from the ESP32. idf_monitor also has a range of features to decode crash output and interact with the device. [Check the documentation page for details](https://docs.espressif.com/projects/esp-idf/en/latest/get-started/idf-monitor.html).
+
+Exit the monitor by typing Ctrl-].
+
+To build, flash and monitor output in one pass, you can run:
+
+`make -j4 flash monitor`
+
+## Compiling & Flashing Only the App
+
+After the initial flash, you may just want to build and flash just your app, not the bootloader and partition table:
+
+* `make app` - build just the app.
+* `make app-flash` - flash just the app.
+
+`make app-flash` will automatically rebuild the app if any source files have changed.
+
+(In normal development there's no downside to reflashing the bootloader and partition table each time, if they haven't changed.)
+
+## Parallel Builds
+
+ESP-IDF supports compiling multiple files in parallel, so all of the above commands can be run as `make -jN` where `N` is the number of parallel make processes to run (generally N should be equal to the number of CPU cores in your system, plus one.)
+
+Multiple make functions can be combined into one. For example: to build the app & bootloader using 5 jobs in parallel, then flash everything, and then display serial output from the ESP32 run:
+
+```
+make -j5 flash monitor
+```
+
+## The Partition Table
+
+Once you've compiled your project, the "build" directory will contain a binary file with a name like "my_app.bin". This is an ESP32 image binary that can be loaded by the bootloader.
+
+A single ESP32's flash can contain multiple apps, as well as many different kinds of data (calibration data, filesystems, parameter storage, etc). For this reason a partition table is flashed to offset 0x8000 in the flash.
+
+Each entry in the partition table has a name (label), type (app, data, or something else), subtype and the offset in flash where the partition is loaded.
+
+The simplest way to use the partition table is to `make menuconfig` and choose one of the simple predefined partition tables:
+
+* "Single factory app, no OTA"
+* "Factory app, two OTA definitions"
+
+In both cases the factory app is flashed at offset 0x10000. If you `make partition_table` then it will print a summary of the partition table.
+
+For more details about partition tables and how to create custom variations, view the [`docs/en/api-guides/partition-tables.rst`](docs/en/api-guides/partition-tables.rst) file.
+
+## Erasing Flash
+
+The `make flash` target does not erase the entire flash contents. However it is sometimes useful to set the device back to a totally erased state, particularly when making partition table changes or OTA app updates. To erase the entire flash, run `make erase_flash`.
+
+This can be combined with other targets, ie `make erase_flash flash` will erase everything and then re-flash the new app, bootloader and partition table.
+
+# Resources
+
+* Documentation for the latest version: https://docs.espressif.com/projects/esp-idf/. This documentation is built from the [docs directory](docs) of this repository.
+
+* The [esp32.com forum](https://esp32.com/) is a place to ask questions and find community resources.
+
+* [Check the Issues section on github](https://github.com/espressif/esp-idf/issues) if you find a bug or have a feature request. Please check existing Issues before opening a new one.
+
+* If you're interested in contributing to ESP-IDF, please check the [Contributions Guide](https://docs.espressif.com/projects/esp-idf/en/latest/contribute/index.html).
+
+
diff --git a/components/bt/Kconfig b/components/bt/Kconfig
index 2215c3e223..9bf12da006 100644
--- a/components/bt/Kconfig
+++ b/components/bt/Kconfig
@@ -1269,3 +1269,782 @@ menu Bluetooth
default 0
endmenu
+
+# Kconfig - Bluetooth Mesh configuration options
+
+#
+# Copyright (c) 2017 Intel Corporation
+#
+# SPDX-License-Identifier: Apache-2.0
+#
+
+menuconfig BT_MESH
+ bool "Bluetooth Mesh support"
+ select TINYCRYPT
+ select TINYCRYPT_AES
+ select TINYCRYPT_AES_CMAC
+ help
+ This option enables Bluetooth Mesh support. The specific
+ features that are available may depend on other features
+ that have been enabled in the stack, such as GATT support.
+
+if BT_MESH
+
+config BT_MESH_FAST_PROV
+ bool "Enable BLE Mesh Fast Provisioning"
+ select BT_MESH_NODE
+ select BT_MESH_PROVISIONER
+ select BT_MESH_PB_ADV
+ default n
+ help
+ Enable this option to allow BLE Mesh fast provisioning solution
+ to be used.
+
+config BT_MESH_NODE
+ bool "Support for BLE Mesh Node"
+ help
+ Enable the device to be provisioned into a node
+
+config BT_MESH_PROVISIONER
+ bool "Support for BLE Mesh Provisioner"
+ help
+ Enable the device to be a provisioner
+
+if BT_MESH_PROVISIONER
+
+config BT_MESH_UNPROV_DEV_ADD
+ int "Maximum number of unprovisioned devices that can be added to device queue"
+ default 20
+ range 1 100
+ help
+ This option specifies how may unprovisioned devices can be added to device queue
+ for provisioning.
+
+config BT_MESH_MAX_STORED_NODES
+ int "Maximum number of nodes whose information can be stored"
+ default 20
+ range 1 1000
+ help
+ This option specifies the maximum number of nodes whose information can be
+ stored by a provisioner in its upper layer.
+
+config BT_MESH_MAX_PROV_NODES
+ int "Maximum number of devices that can be provisioned by provisioner"
+ default 20
+ range 1 100
+ help
+ This option specifies how many devices can be provisioned by
+ provisioner.
+
+if BT_MESH_PB_ADV
+config BT_MESH_PBA_SAME_TIME
+ int "Maximum number of PB-ADV running at the same time by provisioner"
+ default 10
+ range 1 30
+ help
+ This option specifies how many devices can be provisioned at the same
+ time using PB-ADV.
+endif # BT_MESH_PB_ADV
+
+if BT_MESH_PB_GATT
+config BT_MESH_PBG_SAME_TIME
+ int "Maximum number of PB-GATT running at the same time by provisioner"
+ default 4
+ range 1 5
+ help
+ This option specifies how many devices can be provisioned at the same
+ time using PB-GATT.
+endif # BT_MESH_PB_GATT
+
+config BT_MESH_PROVISIONER_SUBNET_COUNT
+ int "Maximum number of mesh subnets that can be created by provisioner"
+ default 3
+ range 1 4096
+ help
+ This option specifies how many subnets per network a provisioner can create
+
+config BT_MESH_PROVISIONER_APP_KEY_COUNT
+ int "Maximum number of application keys that can be owned by provisioner"
+ default 9
+ range 1 4096
+ help
+ This option specifies how many application keys the provisioner
+ can have.
+
+endif # BT_MESH_PROVISIONER
+
+# Virtual option enabled whenever Generic Provisioning layer is needed
+config BT_MESH_PROV
+ bool
+
+config BT_MESH_PB_ADV
+ bool "Provisioning support using the advertising bearer (PB-ADV)"
+ select BT_MESH_PROV
+ default y
+ help
+ Enable this option to allow the device to be provisioned over
+ the advertising bearer.
+
+config BT_MESH_PB_GATT
+ bool "Provisioning support using GATT (PB-GATT)"
+ select BT_MESH_PROXY
+ select BT_MESH_PROV
+ help
+ Enable this option to allow the device to be provisioned over GATT.
+
+# Virtual option enabled whenever any Proxy protocol is needed
+config BT_MESH_PROXY
+ bool
+
+config BT_MESH_GATT_PROXY
+ bool "GATT Proxy Service"
+ select BT_MESH_PROXY
+ help
+ This option enables support for Mesh GATT Proxy Service,
+ i.e. the ability to act as a proxy between a Mesh GATT Client
+ and a Mesh network.
+
+config BT_MESH_NODE_ID_TIMEOUT
+ int "Node Identity advertising timeout"
+ depends on BT_MESH_GATT_PROXY
+ range 1 60
+ default 60
+ help
+ This option determines for how long the local node advertises
+ using Node Identity. The given value is in seconds. The specification
+ limits this to 60 seconds and lists it as the recommended value as well.
+ So leaving the default value is the safest option.
+
+if BT_MESH_PROXY
+
+config BT_MESH_PROXY_FILTER_SIZE
+ int "Maximum number of filter entries per Proxy Client"
+ default 1
+ default 3 if BT_MESH_GATT_PROXY
+ range 1 32767
+ help
+ This option specifies how many Proxy Filter entries the local
+ node supports.
+
+endif # BT_MESH_PROXY
+
+config NET_BUF_POOL_USAGE
+ bool "net buffer pool usage"
+ select BT_MESH_PROV
+ default n
+ help
+ Enable this macro to use the mesh buffer pool usage
+
+config BT_MESH_SETTINGS
+ bool "Store Bluetooth Mesh key and configuration persistently"
+ default n
+ help
+ When selected, the Bluetooth stack will take care of storing/restoring the
+ Bluetooth state(e.g. pairing keys) and configuration persistently in flash.
+
+if BT_MESH_SETTINGS
+config BT_MESH_STORE_TIMEOUT
+ int "Delay (in seconds) before storing anything persistently"
+ range 0 1000000
+ default 2
+ help
+ This value defines in seconds how soon any pending changes
+ are actually written into persistent storage (flash) after
+ a change occurs.
+
+config BT_MESH_SEQ_STORE_RATE
+ int "How often the sequence number gets updated in storage"
+ range 0 1000000
+ default 128
+ help
+ This value defines how often the local sequence number gets
+ updated in persistent storage (i.e. flash). e.g. a value of 100
+ means that the sequence number will be stored to flash on every
+ 100th increment. If the node sends messages very frequently a
+ higher value makes more sense, whereas if the node sends
+ infrequently a value as low as 0 (update storage for every
+ increment) can make sense. When the stack gets initialized it
+ will add sequence number to the last stored one, so that it starts
+ off with a value that's guaranteed to be larger than the last
+ one used before power off.
+
+config BT_MESH_RPL_STORE_TIMEOUT
+ int "Minimum frequency that the RPL gets updated in storage"
+ range 0 1000000
+ default 5
+ help
+ This value defines in seconds how soon the RPL gets written to
+ persistent storage after a change occurs. If the node receives
+ messages frequently, then a large value is recommended. If the
+ node receives messages rarely, then the value can be as low as
+ 0 (which means the PRL is written into the storage immediately).
+ Note that if the node operates in a security-sensitive case, and
+ there is a risk of sudden power-off, then a value of 0 is strongly
+ recommended. Otherwise, a power loss before RPL being written into
+ the storage may introduce message replay attacks and system security
+ will be in a vulnerable state.
+
+config SETTINGS_FCB
+ bool "Settings used fcb mode"
+ default y
+ help
+ Use FCB as a settings storage back-end.
+endif # if BT_MESH_SETTINGS
+
+config BT_MESH_SELF_TEST
+ bool "Perform self-tests"
+ help
+ This option adds extra self-tests which are run every time
+ mesh networking is initialized.
+
+config BT_MESH_IV_UPDATE_TEST
+ bool "Test the IV Update Procedure"
+ help
+ This option removes the 96 hour limit of the IV Update
+ Procedure and lets the state to be changed at any time.
+
+config BT_MESH_SUBNET_COUNT
+ int "Maximum number of mesh subnets per network"
+ default 1
+ range 1 4096
+ help
+ This option specifies how many subnets a Mesh network can
+ have at the same time.
+
+config BT_MESH_APP_KEY_COUNT
+ int "Maximum number of application keys per network"
+ default 1
+ range 1 4096
+ help
+ This option specifies how many application keys the device can
+ store per network.
+
+config BT_MESH_MODEL_KEY_COUNT
+ int "Maximum number of application keys per model"
+ default 1
+ range 1 4096
+ help
+ This option specifies the maximum number of application keys to which
+ each model can be bound.
+
+config BT_MESH_MODEL_GROUP_COUNT
+ int "Maximum number of group address subscriptions per model"
+ default 1
+ range 1 4096
+ help
+ This option specifies the maximum number of addresses to which
+ each model can be subscribed.
+
+config BT_MESH_LABEL_COUNT
+ int "Maximum number of Label UUIDs used for Virtual Addresses"
+ default 1
+ range 0 4096
+ help
+ This option specifies how many Label UUIDs can be stored.
+
+config BT_MESH_CRPL
+ int "Maximum capacity of the replay protection list"
+ default 10
+ range 2 65535
+ help
+ This options specifies the maximum capacity of the replay
+ protection list. It is similar to Network message
+ cache size, but has a different purpose.
+
+config BT_MESH_MSG_CACHE_SIZE
+ int "Network message cache size"
+ default 10
+ range 2 65535
+ help
+ Number of messages that are cached for the network. This helps
+ prevent unnecessary decryption operations and unnecessary
+ relays. This option is similar to Replay protection list,
+ but has a different purpose.
+
+config BT_MESH_ADV_BUF_COUNT
+ int "Number of advertising buffers"
+ default 60
+ range 6 256
+ help
+ Number of advertising buffers available. The transport layer
+ reserves ADV_BUF_COUNT - 3 buffers for outgoing segments. The
+ maximum outgoing SDU size is 12 times this value (out of which
+ 4 or 8 bytes are used for the Transport Layer MIC). For
+ example, 5 segments means the maximum SDU size is 60 bytes,
+ which leaves 56 bytes for application layer data using a
+ 4-byte MIC, or 52 bytes using an 8-byte MIC.
+
+config BT_MESH_TX_SEG_MSG_COUNT
+ int "Maximum number of simultaneous outgoing segmented messages"
+ default 1
+ range 1 BT_MESH_ADV_BUF_COUNT
+ help
+ Maximum number of simultaneous outgoing multi-segment and/or
+ reliable messages.
+
+config BT_MESH_RX_SEG_MSG_COUNT
+ int "Maximum number of simultaneous incoming segmented messages"
+ default 1
+ range 1 255
+ help
+ Maximum number of simultaneous incoming multi-segment and/or
+ reliable messages.
+
+config BT_MESH_RX_SDU_MAX
+ int "Maximum incoming Upper Transport Access PDU length"
+ default 384
+ range 36 384
+ help
+ Maximum incoming Upper Transport Access PDU length. Leave this
+ to the default value, unless you really need to optimize memory
+ usage.
+
+config BT_MESH_RELAY
+ bool "Relay support"
+ help
+ Support for acting as a Mesh Relay Node.
+
+config BT_MESH_LOW_POWER
+ bool "Support for Low Power features"
+ help
+ Enable this option to operate as a Low Power Node.
+
+if BT_MESH_LOW_POWER
+
+config BT_MESH_LPN_ESTABLISHMENT
+ bool "Perform Friendship establishment using low power"
+ default y
+ help
+ Perform the Friendship establishment using low power with
+ the help of a reduced scan duty cycle. The downside of this
+ is that the node may miss out on messages intended for it
+ until it has successfully set up Friendship with a Friend
+ node.
+
+config BT_MESH_LPN_AUTO
+ bool "Automatically start looking for Friend nodes once provisioned"
+ default y
+ help
+ Once provisioned, automatically enable LPN functionality and start
+ looking for Friend nodes. If this option is disabled LPN mode
+ needs to be manually enabled by calling bt_mesh_lpn_set(true).
+
+config BT_MESH_LPN_AUTO_TIMEOUT
+ int "Time from last received message before going to LPN mode"
+ default 15
+ range 0 3600
+ depends on BT_MESH_LPN_AUTO
+ help
+ Time in seconds from the last received message, that the node
+ waits out before starting to look for Friend nodes.
+
+config BT_MESH_LPN_RETRY_TIMEOUT
+ int "Retry timeout for Friend requests"
+ default 8
+ range 1 3600
+ help
+ Time in seconds between Friend Requests, if a previous Friend
+ Request did not yield any acceptable Friend Offers.
+
+config BT_MESH_LPN_RSSI_FACTOR
+ int "RSSIFactor, used in Friend Offer Delay calculation"
+ range 0 3
+ default 0
+ help
+ The contribution of the RSSI, measured by the Friend node, used
+ in Friend Offer Delay calculations. 0 = 1, 1 = 1.5, 2 = 2, 3 = 2.5.
+
+config BT_MESH_LPN_RECV_WIN_FACTOR
+ int "ReceiveWindowFactor, used in Friend Offer Delay calculation"
+ range 0 3
+ default 0
+ help
+ The contribution of the supported Receive Window used in
+ Friend Offer Delay calculations. 0 = 1, 1 = 1.5, 2 = 2, 3 = 2.5.
+
+config BT_MESH_LPN_MIN_QUEUE_SIZE
+ int "Minimum size of the acceptable friend queue (MinQueueSizeLog)"
+ range 1 7
+ default 1
+ help
+ The MinQueueSizeLog field is defined as log_2(N), where N is
+ the minimum number of maximum size Lower Transport PDUs that
+ the Friend node can store in its Friend Queue. As an example,
+ MinQueueSizeLog value 1 gives N = 2, and value 7 gives N = 128.
+
+config BT_MESH_LPN_RECV_DELAY
+ int "Receive delay requested by the local node"
+ range 10 255
+ default 100
+ help
+ The ReceiveDelay is the time between the Low Power node
+ sending a request and listening for a response. This delay
+ allows the Friend node time to prepare the response. The value
+ is in units of milliseconds.
+
+config BT_MESH_LPN_POLL_TIMEOUT
+ int "The value of the PollTimeout timer"
+ range 10 244735
+ default 300
+ help
+ PollTimeout timer is used to measure time between two
+ consecutive requests sent by a Low Power node. If no
+ requests are received by the Friend node before the
+ PollTimeout timer expires, then the friendship is considered
+ terminated. The value is in units of 100 milliseconds, so e.g.
+ a value of 300 means 30 seconds.
+
+config BT_MESH_LPN_INIT_POLL_TIMEOUT
+ int "The starting value of the PollTimeout timer"
+ range 10 BT_MESH_LPN_POLL_TIMEOUT
+ default BT_MESH_LPN_POLL_TIMEOUT
+ help
+ The initial value of the PollTimeout timer when Friendship
+ is to be established for the first time. After this, the timeout
+ gradually grows toward the actual PollTimeout, doubling
+ in value for each iteration. The value is in units of 100
+ milliseconds, so e.g. a value of 300 means 30 seconds.
+
+config BT_MESH_LPN_SCAN_LATENCY
+ int "Latency for enabling scanning"
+ range 0 50
+ default 10
+ help
+ Latency (in milliseconds) is the time it takes to enable scanning. In
+ practice, it means how much time in advance of the Receive Window,
+ the request to enable scanning is made.
+
+config BT_MESH_LPN_GROUPS
+ int "Number of groups the LPN can subscribe to"
+ range 0 16384
+ default 8
+ help
+ Maximum number of groups to which the LPN can subscribe.
+endif # BT_MESH_LOW_POWER
+
+config BT_MESH_FRIEND
+ bool "Support for acting as a Friend Node"
+ help
+ Enable this option to be able to act as a Friend Node.
+
+if BT_MESH_FRIEND
+
+config BT_MESH_FRIEND_RECV_WIN
+ int "Friend Receive Window"
+ range 1 255
+ default 255
+ help
+ Receive Window in milliseconds supported by the Friend node.
+
+config BT_MESH_FRIEND_QUEUE_SIZE
+ int "Minimum number of buffers supported per Friend Queue"
+ range 2 65536
+ default 16
+ help
+ Minimum number of buffers available to be stored for each
+ local Friend Queue.
+
+config BT_MESH_FRIEND_SUB_LIST_SIZE
+ int "Friend Subscription List Size"
+ range 0 1023
+ default 3
+ help
+ Size of the Subscription List that can be supported by a
+ Friend node for a Low Power node.
+
+config BT_MESH_FRIEND_LPN_COUNT
+ int "Number of supported LPN nodes"
+ range 1 1000
+ default 2
+ help
+ Number of Low Power Nodes with which a Friend can have Friendship
+ simultaneously.
+
+config BT_MESH_FRIEND_SEG_RX
+ int "Number of incomplete segment lists per LPN"
+ range 1 1000
+ default 1
+ help
+ Number of incomplete segment lists tracked for each Friends' LPN.
+ In other words, this determines from how many elements can segmented
+ messages destined for the Friend queue be received simultaneously.
+
+endif # BT_MESH_FRIEND
+
+config BT_MESH_NO_LOG
+ bool "Disable BT MESH debug logs (minimize bin size)"
+ depends on BT_MESH
+ default n
+ help
+ Select this to save the rodata code size
+
+menu "BT DEBUG LOG LEVEL"
+ depends on BT_MESH && !BT_MESH_NO_LOG
+
+choice BT_MESH_STACK_INITIAL_TRACE_LEVEL
+ prompt "BT_MESH_STACK"
+ default BT_MESH_TRACE_LEVEL_WARNING
+ depends on BT_MESH && !BT_MESH_NO_LOG
+ help
+ Define BT trace level for bt mesh stack
+
+config BT_MESH_TRACE_LEVEL_NONE
+ bool "NONE"
+config BT_MESH_TRACE_LEVEL_ERROR
+ bool "ERROR"
+config BT_MESH_TRACE_LEVEL_WARNING
+ bool "WARNING"
+config BT_MESH_TRACE_LEVEL_INFO
+ bool "INFO"
+config BT_MESH_TRACE_LEVEL_DEBUG
+ bool "DEBUG"
+config BT_MESH_TRACE_LEVEL_VERBOSE
+ bool "VERBOSE"
+endchoice
+
+config BT_MESH_STACK_INITIAL_TRACE_LEVEL
+ int
+ depends on BT_MESH
+ default 0 if BT_MESH_TRACE_LEVEL_NONE
+ default 1 if BT_MESH_TRACE_LEVEL_ERROR
+ default 2 if BT_MESH_TRACE_LEVEL_WARNING
+ default 3 if BT_MESH_TRACE_LEVEL_INFO
+ default 4 if BT_MESH_TRACE_LEVEL_DEBUG
+ default 5 if BT_MESH_TRACE_LEVEL_VERBOSE
+ default 2
+
+endmenu #BT NET NUFFER DEBUG LOG LEVEL
+
+menu "BT NET NUFFER DEBUG LOG LEVEL"
+ depends on BT_MESH && !BT_MESH_NO_LOG
+
+choice BT_MESH_NET_BUF_INITIAL_TRACE_LEVEL
+ prompt "BT_MESH_STACK"
+ default BT_MESH_NET_BUF_TRACE_LEVEL_WARNING
+ depends on BT_MESH && !BT_MESH_NO_LOG
+ help
+ Define BT trace level for bt mesh network buffer alloc debug
+
+config BT_MESH_NET_BUF_TRACE_LEVEL_NONE
+ bool "NONE"
+config BT_MESH_NET_BUF_TRACE_LEVEL_ERROR
+ bool "ERROR"
+config BT_MESH_NET_BUF_TRACE_LEVEL_WARNING
+ bool "WARNING"
+config BT_MESH_NET_BUF_TRACE_LEVEL_INFO
+ bool "INFO"
+config BT_MESH_NET_BUF_TRACE_LEVEL_DEBUG
+ bool "DEBUG"
+config BT_MESH_NET_BUF_TRACE_LEVEL_VERBOSE
+ bool "VERBOSE"
+endchoice
+
+config BT_MESH_NET_BUF_INITIAL_TRACE_LEVEL
+ int
+ depends on BT_MESH
+ default 0 if BT_MESH_NET_BUF_TRACE_LEVEL_NONE
+ default 1 if BT_MESH_NET_BUF_TRACE_LEVEL_ERROR
+ default 2 if BT_MESH_NET_BUF_TRACE_LEVEL_WARNING
+ default 3 if BT_MESH_NET_BUF_TRACE_LEVEL_INFO
+ default 4 if BT_MESH_NET_BUF_TRACE_LEVEL_DEBUG
+ default 5 if BT_MESH_NET_BUF_TRACE_LEVEL_VERBOSE
+ default 2
+
+endmenu #BT DEBUG NET BUF LOG LEVEL
+
+config BT_MESH_IRQ_LOCK
+ bool "Used the IRQ lock instead of task lock"
+ help
+ To improve the real-time requirements of bt controller in BLE Mesh, task lock is used to replace IRQ lock.
+
+config BT_MESH_CLIENT_MSG_TIMEOUT
+ int "Timeout(ms) for client message response"
+ range 100 1200000
+ default 4000
+ help
+ Timeout value used by the node to get response to the sent message.
+
+config BT_MESH_CFG_CLI
+ bool "Support for Configuration Client Model"
+ help
+ Enable support for Configuration client model.
+
+config BT_MESH_HEALTH_CLI
+ bool "Support for Health Client Model"
+ help
+ Enable support for Health client model.
+
+config BT_MESH_GENERIC_ONOFF_CLI
+ bool "Support for Generic OnOff Client Model"
+ help
+ Enable support for Generic OnOff client model.
+
+config BT_MESH_GENERIC_LEVEL_CLI
+ bool "Support for Generic Level Client Model"
+ help
+ Enable support for Generic Level client model.
+
+config BT_MESH_GENERIC_DEF_TRANS_TIME_CLI
+ bool "Support for Generic Default Transition Time Client Model"
+ help
+ Enable support for Generic Default Transition Time client model.
+
+config BT_MESH_GENERIC_POWER_ONOFF_CLI
+ bool "Support for Generic Power Onoff Client Model"
+ help
+ Enable support for Generic Power Onoff client model.
+
+config BT_MESH_GENERIC_POWER_LEVEL_CLI
+ bool "Support for Generic Power Level Client Model"
+ help
+ Enable support for Generic Power Level client model.
+
+config BT_MESH_GENERIC_BATTERY_CLI
+ bool "Support for Generic Battery Client Model"
+ help
+ Enable support for Generic Battery client model.
+
+config BT_MESH_GENERIC_LOCATION_CLI
+ bool "Support for Generic Location Client Model"
+ help
+ Enable support for Generic Location client model.
+
+config BT_MESH_GENERIC_PROPERTY_CLI
+ bool "Support for Generic Property Client Model"
+ help
+ Enable support for Generic Property client model.
+
+config BT_MESH_SENSOR_CLI
+ bool "Support for Sensor Client Model"
+ help
+ Enable support for Sensor client model.
+
+config BT_MESH_TIME_CLI
+ bool "Support for Time Client Model"
+ help
+ Enable support for Time client model.
+
+config BT_MESH_SCENE_CLI
+ bool "Support for Scene Client Model"
+ help
+ Enable support for Scene client model.
+
+config BT_MESH_SCHEDULER_CLI
+ bool "Support for Scheduler Client Model"
+ help
+ Enable support for Scheduler client model.
+
+config BT_MESH_LIGHT_LIGHTNESS_CLI
+ bool "Support for Light Lightness Client Model"
+ help
+ Enable support for Light Lightness client model.
+
+config BT_MESH_LIGHT_CTL_CLI
+ bool "Support for Light CTL Client Model"
+ help
+ Enable support for Light CTL client model.
+
+config BT_MESH_LIGHT_HSL_CLI
+ bool "Support for Light HSL Client Model"
+ help
+ Enable support for Light HSL client model.
+
+config BT_MESH_LIGHT_XYL_CLI
+ bool "Support for Light XYL Client Model"
+ help
+ Enable support for Light XYL client model.
+
+config BT_MESH_LIGHT_LC_CLI
+ bool "Support for Light LC Client Model"
+ help
+ Enable support for Light LC client model.
+
+config BT_MESH_SHELL
+ bool "Enable Bluetooth Mesh shell"
+ select CONSOLE_SHELL
+ depends on BT_MESH_CFG_CLI
+ depends on BT_MESH_HEALTH_CLI
+ help
+ Activate shell module that provides Bluetooth Mesh commands to
+ the console.
+
+
+
+config BT_MESH_DEBUG
+ bool "Enable debug logs"
+ depends on BT_DEBUG
+ help
+ Use this option to enable debug logs for the Bluetooth
+ Mesh functionality.
+
+if BT_MESH_DEBUG
+
+config BT_MESH_DEBUG_NET
+ bool "Network layer debug"
+ help
+ Use this option to enable Network layer debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_TRANS
+ bool "Transport layer debug"
+ help
+ Use this option to enable Transport layer debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_BEACON
+ bool "Beacon debug"
+ help
+ Use this option to enable Beacon-related debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_CRYPTO
+ bool "Crypto debug"
+ help
+ Use this option to enable cryptographic debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_PROV
+ bool "Provisioning debug"
+ help
+ Use this option to enable Provisioning debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_ACCESS
+ bool "Access layer debug"
+ help
+ Use this option to enable Access layer and device composition
+ related debug logs for Bluetooth Mesh.
+
+config BT_MESH_DEBUG_MODEL
+ bool "Foundation model debug"
+ help
+ Use this option to enable debug logs for the Foundation
+ Models.
+
+config BT_MESH_DEBUG_ADV
+ bool "Advertising debug"
+ help
+ Use this option to enable advertising debug logs for
+ the Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_LOW_POWER
+ bool "Low Power debug"
+ help
+ Use this option to enable Low Power debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_FRIEND
+ bool "Friend debug"
+ help
+ Use this option to enable Friend debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_PROXY
+ bool "Proxy debug"
+ depends on BT_MESH_PROXY
+ help
+ Use this option to enable Proxy protocol debug logs.
+
+endif # BT_MESH_DEBUG
+
+endif # BT_MESH
diff --git a/components/bt/ble_mesh/api/core/esp_ble_mesh_common_api.c b/components/bt/ble_mesh/api/core/esp_ble_mesh_common_api.c
new file mode 100644
index 0000000000..33019c6e60
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/esp_ble_mesh_common_api.c
@@ -0,0 +1,77 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "freertos/queue.h"
+#include "freertos/semphr.h"
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "esp_bt_defs.h"
+#include "esp_err.h"
+#include "esp_bt_main.h"
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_prov.h"
+
+#include "mesh.h"
+#include "mesh_buf.h"
+#include "transport.h"
+#include "esp_ble_mesh_common_api.h"
+
+#if CONFIG_BT_MESH
+
+esp_err_t esp_ble_mesh_init(esp_ble_mesh_prov_t *prov, esp_ble_mesh_comp_t *comp)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg = {0};
+ xSemaphoreHandle semaphore = NULL;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ // Create a semaphore
+ if ((semaphore = xSemaphoreCreateCounting(1, 0)) == NULL) {
+ LOG_ERROR("%s, unable to allocate memory for the semaphore.", __func__);
+ return ESP_ERR_NO_MEM;
+ }
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_APP_REGISTER;
+ arg.mesh_reg.prov = prov;
+ arg.mesh_reg.comp = comp;
+ // semaphore pointer transport to BTC layer, we will give the semaphore in the BTC task.
+ arg.mesh_reg.semaphore = semaphore;
+
+ if (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL) != BT_STATUS_SUCCESS) {
+ vSemaphoreDelete(semaphore);
+ LOG_ERROR("BLE Mesh initialise failed");
+ return ESP_FAIL;
+ }
+
+ // Take the Semaphore, wait to BLE Mesh init finish.
+ xSemaphoreTake(semaphore, portMAX_DELAY);
+ // Don't forget to delete the semaphore at the end.
+ vSemaphoreDelete(semaphore);
+ return ESP_OK;
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/api/core/esp_ble_mesh_local_data_operation_api.c b/components/bt/ble_mesh/api/core/esp_ble_mesh_local_data_operation_api.c
new file mode 100644
index 0000000000..d036f303f7
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/esp_ble_mesh_local_data_operation_api.c
@@ -0,0 +1,90 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "esp_bt_defs.h"
+#include "esp_err.h"
+#include "esp_bt_main.h"
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_prov.h"
+
+#include "mesh.h"
+#include "mesh_buf.h"
+#include "transport.h"
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+int32_t esp_ble_mesh_get_model_publish_period(esp_ble_mesh_model_t *model)
+{
+ if (model == NULL) {
+ return -EINVAL;
+ }
+ return btc_ble_mesh_model_pub_period_get(model);
+}
+
+uint16_t esp_ble_mesh_get_primary_element_address(void)
+{
+ return btc_ble_mesh_get_primary_addr();
+}
+
+uint16_t *esp_ble_mesh_is_model_subscribed_to_group(esp_ble_mesh_model_t *model, uint16_t group_addr)
+{
+ if (model == NULL) {
+ return NULL;
+ }
+ return btc_ble_mesh_model_find_group(model, group_addr);
+}
+
+esp_ble_mesh_elem_t *esp_ble_mesh_find_element(uint16_t element_addr)
+{
+ return btc_ble_mesh_elem_find(element_addr);
+}
+
+uint8_t esp_ble_mesh_get_element_count(void)
+{
+ return btc_ble_mesh_elem_count();
+}
+
+esp_ble_mesh_model_t *esp_ble_mesh_find_vendor_model(esp_ble_mesh_elem_t *element,
+ uint16_t company_id, uint16_t model_id)
+{
+ if (element == NULL) {
+ return NULL;
+ }
+ return btc_ble_mesh_model_find_vnd(element, company_id, model_id);
+}
+
+esp_ble_mesh_model_t *esp_ble_mesh_find_sig_model(esp_ble_mesh_elem_t *element, uint16_t model_id)
+{
+ if (element == NULL) {
+ return NULL;
+ }
+ return btc_ble_mesh_model_find(element, model_id);
+}
+
+const esp_ble_mesh_comp_t *esp_ble_mesh_get_composition_data(void)
+{
+ return btc_ble_mesh_comp_get();
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/api/core/esp_ble_mesh_low_power_api.c b/components/bt/ble_mesh/api/core/esp_ble_mesh_low_power_api.c
new file mode 100644
index 0000000000..14158ec4d5
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/esp_ble_mesh_low_power_api.c
@@ -0,0 +1,37 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "esp_bt_defs.h"
+#include "esp_err.h"
+#include "esp_bt_main.h"
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_prov.h"
+
+#include "mesh.h"
+#include "mesh_buf.h"
+#include "transport.h"
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/api/core/esp_ble_mesh_networking_api.c b/components/bt/ble_mesh/api/core/esp_ble_mesh_networking_api.c
new file mode 100644
index 0000000000..f3ad8df547
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/esp_ble_mesh_networking_api.c
@@ -0,0 +1,344 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "esp_bt_defs.h"
+#include "esp_err.h"
+#include "esp_bt_main.h"
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_prov.h"
+
+#include "mesh.h"
+#include "mesh_buf.h"
+#include "transport.h"
+#include "provisioner_main.h"
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_networking_api.h"
+
+#if CONFIG_BT_MESH
+
+static esp_err_t ble_mesh_send_msg(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx,
+ uint32_t opcode,
+ btc_ble_mesh_model_act_t act,
+ uint16_t length, uint8_t *data,
+ int32_t msg_timeout, bool need_rsp,
+ esp_ble_mesh_dev_role_t device_role)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_model_args_t arg;
+ esp_err_t status;
+ uint8_t op_len = 0, mic_len = 0;
+ uint8_t *msg_data = NULL;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ if (device_role > ROLE_FAST_PROV) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ /* When data is NULL, it is mandatory to set length to 0 to prevent users from misinterpreting parameters. */
+ if (data == NULL) {
+ length = 0;
+ }
+
+ if (opcode < 0x100) {
+ op_len = 1;
+ } else if (opcode < 0x10000) {
+ op_len = 2;
+ } else {
+ op_len = 3;
+ }
+
+ if (act == BTC_BLE_MESH_ACT_MODEL_PUBLISH) {
+ if (op_len + length > model->pub->msg->size) {
+ LOG_ERROR("%s: model->pub->msg->size %d is too small", __func__, model->pub->msg->size);
+ return ESP_ERR_INVALID_ARG;
+ }
+ }
+
+ if (act == BTC_BLE_MESH_ACT_MODEL_PUBLISH) {
+ mic_len = 4;
+ } else {
+ mic_len = ctx->send_rel ? 8 : 4;
+ }
+
+ if (op_len + length + mic_len > MIN(ESP_BLE_MESH_SDU_MAX_LEN, BT_MESH_TX_SDU_MAX)) {
+ LOG_ERROR("%s: The data length %d is too large", __func__, length);
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ if (act == BTC_BLE_MESH_ACT_MODEL_PUBLISH) {
+ bt_mesh_model_msg_init(model->pub->msg, opcode);
+ net_buf_simple_add_mem(model->pub->msg, data, length);
+ } else {
+ msg_data = (uint8_t *)osi_malloc(op_len + length);
+ if (msg_data == NULL) {
+ return ESP_ERR_NO_MEM;
+ }
+ esp_ble_mesh_model_msg_opcode_init(msg_data, opcode);
+ memcpy(msg_data + op_len, data, length);
+ }
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_MODEL;
+ msg.act = act;
+
+ if (act == BTC_BLE_MESH_ACT_MODEL_PUBLISH) {
+ arg.model_publish.model = model;
+ arg.model_publish.device_role = device_role;
+ } else {
+ arg.model_send.model = model;
+ arg.model_send.ctx = ctx;
+ arg.model_send.need_rsp = need_rsp;
+ arg.model_send.opcode = opcode;
+ arg.model_send.length = op_len + length;
+ arg.model_send.data = msg_data;
+ arg.model_send.device_role = device_role;
+ arg.model_send.msg_timeout = msg_timeout;
+ }
+
+ status = (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_model_args_t), btc_ble_mesh_prov_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+
+ osi_free(msg_data);
+
+ return status;
+}
+
+esp_err_t esp_ble_mesh_register_custom_model_callback(esp_ble_mesh_model_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_MODEL, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_model_msg_opcode_init(uint8_t *data, uint32_t opcode)
+{
+ uint16_t val;
+
+ if (data == NULL) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ if (opcode < 0x100) {
+ /* 1-byte OpCode */
+ data[0] = opcode & 0xff;
+ return ESP_OK;
+ }
+
+ if (opcode < 0x10000) {
+ /* 2-byte OpCode, big endian */
+ val = sys_cpu_to_be16 (opcode);
+ memcpy (data, &val, 2);
+ return ESP_OK;
+ }
+
+ /* 3-byte OpCode, note that little endian for the least 2 bytes(Company ID) of opcode */
+ data[0] = (opcode >> 16) & 0xff;
+ val = sys_cpu_to_le16(opcode & 0xffff);
+ memcpy (&data[1], &val, 2);
+ return ESP_OK;
+}
+
+int esp_ble_mesh_client_model_init(esp_ble_mesh_model_t *model)
+{
+ return btc_ble_mesh_client_init(model);
+}
+
+esp_err_t esp_ble_mesh_server_model_send_msg(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx, uint32_t opcode,
+ uint16_t length, uint8_t *data)
+{
+ if (!model || !ctx) {
+ return ESP_ERR_INVALID_ARG;
+ }
+ return ble_mesh_send_msg(model, ctx, opcode, BTC_BLE_MESH_ACT_SERVER_MODEL_SEND,
+ length, data, 0, false, ROLE_NODE);
+}
+
+esp_err_t esp_ble_mesh_client_model_send_msg(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx, uint32_t opcode,
+ uint16_t length, uint8_t *data, int32_t msg_timeout,
+ bool need_rsp, esp_ble_mesh_dev_role_t device_role)
+{
+ if (!model || !ctx) {
+ return ESP_ERR_INVALID_ARG;
+ }
+ return ble_mesh_send_msg(model, ctx, opcode, BTC_BLE_MESH_ACT_CLIENT_MODEL_SEND,
+ length, data, msg_timeout, need_rsp, device_role);
+}
+
+esp_err_t esp_ble_mesh_model_publish(esp_ble_mesh_model_t *model, uint32_t opcode,
+ uint16_t length, uint8_t *data,
+ esp_ble_mesh_dev_role_t device_role)
+{
+ if (!model || !model->pub || !model->pub->msg) {
+ return ESP_ERR_INVALID_ARG;
+ }
+ return ble_mesh_send_msg(model, NULL, opcode, BTC_BLE_MESH_ACT_MODEL_PUBLISH,
+ length, data, 0, false, device_role);
+}
+
+esp_err_t esp_ble_mesh_node_local_reset(void)
+{
+ btc_msg_t msg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_NODE_RESET;
+
+ return (btc_transfer_context(&msg, NULL, 0, NULL) == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+#if (CONFIG_BT_MESH_PROVISIONER)
+
+esp_err_t esp_ble_mesh_provisioner_set_node_name(int index, const char *name)
+{
+ if (!name || (strlen(name) > ESP_BLE_MESH_NODE_NAME_MAX_LEN)) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_SET_NODE_NAME;
+
+ arg.set_node_name.index = index;
+ memset(arg.set_node_name.name, 0, sizeof(arg.set_node_name.name));
+ memcpy(arg.set_node_name.name, name, strlen(name));
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+const char *esp_ble_mesh_provisioner_get_node_name(int index)
+{
+ return bt_mesh_provisioner_get_node_name(index);
+}
+
+int esp_ble_mesh_provisioner_get_node_index(const char *name)
+{
+ if (!name || (strlen(name) > ESP_BLE_MESH_NODE_NAME_MAX_LEN)) {
+ return -EINVAL;
+ }
+
+ return bt_mesh_provisioner_get_node_index(name);
+}
+
+esp_err_t esp_ble_mesh_provisioner_add_local_app_key(const uint8_t app_key[16],
+ uint16_t net_idx, uint16_t app_idx)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_SET_LOCAL_APP_KEY;
+
+ arg.add_local_app_key.net_idx = net_idx;
+ arg.add_local_app_key.app_idx = app_idx;
+ if (app_key) {
+ memcpy(arg.add_local_app_key.app_key, app_key, 16);
+ } else {
+ bzero(arg.add_local_app_key.app_key, 16);
+ }
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+const uint8_t *esp_ble_mesh_provisioner_get_local_app_key(uint16_t net_idx, uint16_t app_idx)
+{
+ return bt_mesh_provisioner_local_app_key_get(net_idx, app_idx);
+}
+
+esp_err_t esp_ble_mesh_provisioner_bind_app_key_to_local_model(uint16_t element_addr, uint16_t app_idx,
+ uint16_t model_id, uint16_t company_id)
+{
+ if (!ESP_BLE_MESH_ADDR_IS_UNICAST(element_addr)) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_BIND_LOCAL_MOD_APP;
+
+ arg.local_mod_app_bind.elem_addr = element_addr;
+ arg.local_mod_app_bind.app_idx = app_idx;
+ arg.local_mod_app_bind.model_id = model_id;
+ arg.local_mod_app_bind.cid = company_id;
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_add_local_net_key(const uint8_t net_key[16], uint16_t net_idx)
+{
+ if (net_idx == ESP_BLE_MESH_KEY_PRIMARY) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_ADD_LOCAL_NET_KEY;
+
+ arg.add_local_net_key.net_idx = net_idx;
+ if (net_key) {
+ memcpy(arg.add_local_net_key.net_key, net_key, 16);
+ } else {
+ bzero(arg.add_local_net_key.net_key, 16);
+ }
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+const uint8_t *esp_ble_mesh_provisioner_get_local_net_key(uint16_t net_idx)
+{
+ return bt_mesh_provisioner_local_net_key_get(net_idx);
+}
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+#if (CONFIG_BT_MESH_FAST_PROV)
+const uint8_t *esp_ble_mesh_get_fast_prov_app_key(uint16_t net_idx, uint16_t app_idx)
+{
+ return bt_mesh_get_fast_prov_app_key(net_idx, app_idx);
+}
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/api/core/esp_ble_mesh_provisioning_api.c b/components/bt/ble_mesh/api/core/esp_ble_mesh_provisioning_api.c
new file mode 100644
index 0000000000..42b8f55c4d
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/esp_ble_mesh_provisioning_api.c
@@ -0,0 +1,436 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "esp_bt_defs.h"
+#include "esp_err.h"
+#include "esp_bt_main.h"
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_prov.h"
+
+#include "mesh.h"
+#include "mesh_buf.h"
+#include "transport.h"
+#include "provisioner_prov.h"
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_provisioning_api.h"
+
+#if CONFIG_BT_MESH
+
+#define MAX_PROV_LINK_IDX (CONFIG_BT_MESH_PBA_SAME_TIME + CONFIG_BT_MESH_PBG_SAME_TIME)
+#define MAX_OOB_INPUT_NUM 0x5F5E0FF /* Decimal: 99999999 */
+
+esp_err_t esp_ble_mesh_register_prov_callback(esp_ble_mesh_prov_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_PROV, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+bool esp_ble_mesh_node_is_provisioned(void)
+{
+ return bt_mesh_is_provisioned();
+}
+
+esp_err_t esp_ble_mesh_node_prov_enable(esp_ble_mesh_prov_bearer_t bearers)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROV_ENABLE;
+ arg.mesh_prov_enable.bearers = bearers;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_node_prov_disable(esp_ble_mesh_prov_bearer_t bearers)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROV_DISABLE;
+ arg.mesh_prov_disable.bearers = bearers;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_node_set_oob_pub_key(uint8_t pub_key_x[32], uint8_t pub_key_y[32],
+ uint8_t private_key[32])
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (!pub_key_x || !pub_key_y || !private_key) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_SET_OOB_PUB_KEY;
+
+ memcpy(arg.set_oob_pub_key.pub_key_x, pub_key_x, 32);
+ memcpy(arg.set_oob_pub_key.pub_key_y, pub_key_y, 32);
+ memcpy(arg.set_oob_pub_key.private_key, private_key, 32);
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_node_input_number(uint32_t number)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (number > MAX_OOB_INPUT_NUM) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_INPUT_NUMBER;
+ arg.input_num.number = number;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_node_input_string(const char *string)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (!string) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_INPUT_STRING;
+ memset(arg.input_str.string, 0, sizeof(arg.input_str.string));
+ strncpy(arg.input_str.string, string, strlen(string));
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_set_unprovisioned_device_name(const char *name)
+{
+ if (!name || strlen(name) > ESP_BLE_MESH_DEVICE_NAME_MAX_LEN) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_SET_DEVICE_NAME;
+
+ memset(arg.set_device_name.name, 0, sizeof(arg.set_device_name.name));
+ memcpy(arg.set_device_name.name, name, strlen(name));
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+#if (CONFIG_BT_MESH_PROVISIONER)
+esp_err_t esp_ble_mesh_provisioner_read_oob_pub_key(uint8_t link_idx, uint8_t pub_key_x[32],
+ uint8_t pub_key_y[32])
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (!pub_key_x || !pub_key_y || link_idx >= MAX_PROV_LINK_IDX) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_READ_OOB_PUB_KEY;
+
+ arg.provisioner_read_oob_pub_key.link_idx = link_idx;
+ memcpy(arg.provisioner_read_oob_pub_key.pub_key_x, pub_key_x, 32);
+ memcpy(arg.provisioner_read_oob_pub_key.pub_key_y, pub_key_y, 32);
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_input_string(const char *string, uint8_t link_idx)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (!string || link_idx >= MAX_PROV_LINK_IDX) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_INPUT_STR;
+
+ memset(arg.provisioner_input_str.string, 0, sizeof(arg.provisioner_input_str.string));
+ strncpy(arg.provisioner_input_str.string, string, strlen(string));
+ arg.provisioner_input_str.link_idx = link_idx;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_input_number(uint32_t number, uint8_t link_idx)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (number > MAX_OOB_INPUT_NUM || link_idx >= MAX_PROV_LINK_IDX) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_INPUT_NUM;
+
+ arg.provisioner_input_num.number = number;
+ arg.provisioner_input_num.link_idx = link_idx;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_prov_enable(esp_ble_mesh_prov_bearer_t bearers)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_ENABLE;
+
+ arg.provisioner_enable.bearers = bearers;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_prov_disable(esp_ble_mesh_prov_bearer_t bearers)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_DISABLE;
+
+ arg.provisioner_disable.bearers = bearers;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_add_unprov_dev(esp_ble_mesh_unprov_dev_add_t *add_dev,
+ esp_ble_mesh_dev_add_flag_t flags)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (add_dev == NULL) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_DEV_ADD;
+
+ arg.provisioner_dev_add.add_dev.addr_type = add_dev->addr_type;
+ arg.provisioner_dev_add.add_dev.oob_info = add_dev->oob_info;
+ arg.provisioner_dev_add.add_dev.bearer = add_dev->bearer;
+ memcpy(arg.provisioner_dev_add.add_dev.addr, add_dev->addr, sizeof(esp_bd_addr_t));
+ memcpy(arg.provisioner_dev_add.add_dev.uuid, add_dev->uuid, 16);
+ arg.provisioner_dev_add.flags = flags;
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_delete_dev(esp_ble_mesh_device_delete_t *del_dev)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+ uint8_t val = DEL_DEV_ADDR_FLAG | DEL_DEV_UUID_FLAG;
+
+ if (del_dev == NULL || (__builtin_popcount(del_dev->flag & val) != 1)) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_DEV_DEL;
+
+ arg.provisioner_dev_del.del_dev.flag = del_dev->flag;
+ if (del_dev->flag & DEL_DEV_ADDR_FLAG) {
+ arg.provisioner_dev_del.del_dev.addr_type = del_dev->addr_type;
+ memcpy(arg.provisioner_dev_del.del_dev.addr, del_dev->addr, sizeof(esp_bd_addr_t));
+ } else if (del_dev->flag & DEL_DEV_UUID_FLAG) {
+ memcpy(arg.provisioner_dev_del.del_dev.uuid, del_dev->uuid, 16);
+ }
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_set_dev_uuid_match(const uint8_t *match_val, uint8_t match_len,
+ uint8_t offset, bool prov_after_match)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_SET_DEV_UUID_MATCH;
+
+ if (match_len && match_val) {
+ memcpy(arg.set_dev_uuid_match.match_val, match_val, match_len);
+ }
+ arg.set_dev_uuid_match.match_len = match_len;
+ arg.set_dev_uuid_match.offset = offset;
+ arg.set_dev_uuid_match.prov_after_match = prov_after_match;
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_provisioner_set_prov_data_info(esp_ble_mesh_prov_data_info_t *prov_data_info)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+ uint8_t val = PROV_DATA_NET_IDX_FLAG | PROV_DATA_FLAGS_FLAG | PROV_DATA_IV_INDEX_FLAG;
+
+ if (prov_data_info == NULL || (__builtin_popcount(prov_data_info->flag & val) != 1)) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROVISIONER_SET_PROV_DATA_INFO;
+
+ arg.set_prov_data_info.prov_data.flag = prov_data_info->flag;
+ if (prov_data_info->flag & PROV_DATA_NET_IDX_FLAG) {
+ arg.set_prov_data_info.prov_data.net_idx = prov_data_info->net_idx;
+ } else if (prov_data_info->flag & PROV_DATA_FLAGS_FLAG) {
+ arg.set_prov_data_info.prov_data.flags = prov_data_info->flags;
+ } else if (prov_data_info->flag & PROV_DATA_IV_INDEX_FLAG) {
+ arg.set_prov_data_info.prov_data.iv_index = prov_data_info->iv_index;
+ }
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+/* The following APIs are for fast provisioning */
+
+#if (CONFIG_BT_MESH_FAST_PROV)
+
+esp_err_t esp_ble_mesh_set_fast_prov_info(esp_ble_mesh_fast_prov_info_t *fast_prov_info)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (fast_prov_info == NULL) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_SET_FAST_PROV_INFO;
+
+ arg.set_fast_prov_info.unicast_min = fast_prov_info->unicast_min;
+ arg.set_fast_prov_info.unicast_max = fast_prov_info->unicast_max;
+ arg.set_fast_prov_info.net_idx = fast_prov_info->net_idx;
+ arg.set_fast_prov_info.flags = fast_prov_info->flags;
+ arg.set_fast_prov_info.iv_index = fast_prov_info->iv_index;
+ arg.set_fast_prov_info.offset = fast_prov_info->offset;
+ arg.set_fast_prov_info.match_len = fast_prov_info->match_len;
+ if (fast_prov_info->match_len && fast_prov_info->match_val) {
+ memcpy(arg.set_fast_prov_info.match_val, fast_prov_info->match_val, fast_prov_info->match_len);
+ }
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_set_fast_prov_action(esp_ble_mesh_fast_prov_action_t action)
+{
+ btc_msg_t msg;
+ btc_ble_mesh_prov_args_t arg;
+
+ if (action >= FAST_PROV_ACT_MAX) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_SET_FAST_PROV_ACTION;
+
+ arg.set_fast_prov_action.action = action;
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_prov_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/api/core/esp_ble_mesh_proxy_api.c b/components/bt/ble_mesh/api/core/esp_ble_mesh_proxy_api.c
new file mode 100644
index 0000000000..3684a11a67
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/esp_ble_mesh_proxy_api.c
@@ -0,0 +1,75 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "esp_bt_defs.h"
+#include "esp_err.h"
+#include "esp_bt_main.h"
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_prov.h"
+
+#include "mesh.h"
+#include "mesh_buf.h"
+#include "transport.h"
+
+#if CONFIG_BT_MESH
+
+esp_err_t esp_ble_mesh_proxy_identity_enable(void)
+{
+ btc_msg_t msg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROXY_IDENTITY_ENABLE;
+
+ return (btc_transfer_context(&msg, NULL, 0, NULL) == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_proxy_gatt_enable(void)
+{
+ btc_msg_t msg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROXY_GATT_ENABLE;
+
+ return (btc_transfer_context(&msg, NULL, 0, NULL) == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_proxy_gatt_disable(void)
+{
+ btc_msg_t msg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_PROV;
+ msg.act = BTC_BLE_MESH_ACT_PROXY_GATT_DISABLE;
+
+ return (btc_transfer_context(&msg, NULL, 0, NULL) == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/api/core/include/esp_ble_mesh_common_api.h b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_common_api.h
new file mode 100644
index 0000000000..00a183e2cf
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_common_api.h
@@ -0,0 +1,39 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_COMMON_API_H_
+#define _ESP_BLE_MESH_COMMON_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+/**
+ * @brief Initialize BLE Mesh module.
+ * This API initializes provisioning capabilities and composition data information.
+ *
+ * @note After calling this API, the device needs to call esp_ble_mesh_prov_enable()
+ * to enable provisioning functionality again.
+ *
+ * @param[in] prov: Pointer to the device provisioning capabilities.
+ * @param[in] comp: Pointer to the device composition data information.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_init(esp_ble_mesh_prov_t *prov, esp_ble_mesh_comp_t *comp);
+
+#endif /* #if CONFIG_BT_MESH */
+#endif /* _ESP_BLE_MESH_COMMON_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/core/include/esp_ble_mesh_local_data_operation_api.h b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_local_data_operation_api.h
new file mode 100644
index 0000000000..e060ce2ccc
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_local_data_operation_api.h
@@ -0,0 +1,117 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_LOCAL_DATA_OPERATION_API_H_
+#define _ESP_BLE_MESH_LOCAL_DATA_OPERATION_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+/**
+ * @brief Get the model publish period, the unit is ms.
+ *
+ * @param[in] model: Model instance pointer.
+ *
+ * @return Publish period value on success, 0 or (negative) error code from errno.h on failure.
+ *
+ */
+int32_t esp_ble_mesh_get_model_publish_period(esp_ble_mesh_model_t *model);
+
+/**
+ * @brief Get the address of the primary element.
+ *
+ * @param None.
+ *
+ * @return Address of the primary element on success, or
+ * ESP_BLE_MESH_ADDR_UNASSIGNED on failure which means the device has not been provisioned.
+ *
+ */
+uint16_t esp_ble_mesh_get_primary_element_address(void);
+
+/**
+ * @brief Check if the model has subscribed to the given group address.
+ * Note: E.g., once a status message is received and the destination address
+ * is a group address, the model uses this API to check if it is successfully subscribed
+ * to the given group address.
+ *
+ * @param[in] model: Pointer to the model.
+ * @param[in] group_addr: Group address.
+ *
+ * @return Pointer to the group address within the Subscription List of the model on success, or
+ * NULL on failure which means the model has not subscribed to the given group address.
+ * Note: With the pointer to the group address returned, you can reset the group address
+ * to 0x0000 in order to unsubscribe the model from the group.
+ *
+ */
+uint16_t *esp_ble_mesh_is_model_subscribed_to_group(esp_ble_mesh_model_t *model, uint16_t group_addr);
+
+/**
+ * @brief Find the BLE Mesh element pointer via the element address.
+ *
+ * @param[in] element_addr: Element address.
+ *
+ * @return Pointer to the element on success, or NULL on failure.
+ *
+ */
+esp_ble_mesh_elem_t *esp_ble_mesh_find_element(uint16_t element_addr);
+
+/**
+ * @brief Get the number of elements that have been registered.
+ *
+ * @param None.
+ *
+ * @return Number of elements.
+ *
+ */
+uint8_t esp_ble_mesh_get_element_count(void);
+
+/**
+ * @brief Find the Vendor specific model with the given element,
+ * the company ID and the Vendor Model ID.
+ *
+ * @param[in] element: Element to which the model belongs.
+ * @param[in] company_id: A 16-bit company identifier assigned by the Bluetooth SIG.
+ * @param[in] model_id: A 16-bit vendor-assigned model identifier.
+ *
+ * @return Pointer to the Vendor Model on success, or NULL on failure which means the Vendor Model is not found.
+ *
+ */
+esp_ble_mesh_model_t *esp_ble_mesh_find_vendor_model(esp_ble_mesh_elem_t *element,
+ uint16_t company_id, uint16_t model_id);
+
+/**
+ * @brief Find the SIG model with the given element and Model id.
+ *
+ * @param[in] element: Element to which the model belongs.
+ * @param[in] model_id: SIG model identifier.
+ *
+ * @return Pointer to the SIG Model on success, or NULL on failure which means the SIG Model is not found.
+ *
+ */
+esp_ble_mesh_model_t *esp_ble_mesh_find_sig_model(esp_ble_mesh_elem_t *element, uint16_t model_id);
+
+/**
+ * @brief Get the Composition data which has been registered.
+ *
+ * @param None.
+ *
+ * @return Pointer to the Composition data on success, or NULL on failure which means the Composition data is not initialized.
+ *
+ */
+const esp_ble_mesh_comp_t *esp_ble_mesh_get_composition_data(void);
+
+#endif /* #if CONFIG_BT_MESH */
+#endif /* _ESP_BLE_MESH_LOCAL_DATA_OPERATION_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/core/include/esp_ble_mesh_low_power_api.h b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_low_power_api.h
new file mode 100644
index 0000000000..485298bfe3
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_low_power_api.h
@@ -0,0 +1,24 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_LOW_POWER_API_H_
+#define _ESP_BLE_MESH_LOW_POWER_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+#endif /* #if CONFIG_BT_MESH */
+#endif /* _ESP_BLE_MESH_LOW_POWER_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/core/include/esp_ble_mesh_networking_api.h b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_networking_api.h
new file mode 100644
index 0000000000..5b8ee9d0bc
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_networking_api.h
@@ -0,0 +1,274 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_NETWORKING_API_H_
+#define _ESP_BLE_MESH_NETWORKING_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+/** @brief: event, event code of user-defined model events; param, parameters of user-defined model events */
+typedef void (* esp_ble_mesh_model_cb_t)(esp_ble_mesh_model_cb_event_t event,
+ esp_ble_mesh_model_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh callback for user-defined models' operations.
+ * This callback can report the following events generated for the user-defined models:
+ * - Call back the messages received by user-defined client and server models to the
+ * application layer;
+ * - If users call esp_ble_mesh_server/client_model_send, this callback notifies the
+ * application layer of the send_complete event;
+ * - If user-defined client model sends a message that requires response, and the response
+ * message is received after the timer expires, the response message will be reported
+ * to the application layer as published by a peer device;
+ * - If the user-defined client model fails to receive the response message during a specified
+ * period of time, a timeout event will be reported to the application layer.
+ *
+ * @note The client models (i.e. Config Client model, Health Client model, Generic
+ * Client models, Sensor Client model, Scene Client model and Lighting Client models)
+ * that have been realized internally have their specific register functions.
+ * For example, esp_ble_mesh_register_config_client_callback is the register
+ * function for Config Client Model.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_custom_model_callback(esp_ble_mesh_model_cb_t callback);
+
+/**
+ * @brief Add the message opcode to the beginning of the model message
+ * before sending or publishing the model message.
+ *
+ * @note This API is only used to set the opcode of the message.
+ *
+ * @param[in] data: Pointer to the message data.
+ * @param[in] opcode: The message opcode.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_model_msg_opcode_init(uint8_t *data, uint32_t opcode);
+
+/**
+ * @brief Initialize the user-defined client model. All user-defined client models
+ * shall call this function to initialize the client model internal data.
+ * Node: Before calling this API, the op_pair_size and op_pair variabled within
+ * the user_data(defined using esp_ble_mesh_client_t_) of the client model
+ * need to be initialized.
+ *
+ * @param[in] model: BLE Mesh Client model to which the message belongs.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_client_model_init(esp_ble_mesh_model_t *model);
+
+/**
+ * @brief Send server model messages(such as server model status messages).
+ *
+ * @param[in] model: BLE Mesh Server Model to which the message belongs.
+ * @param[in] ctx: Message context, includes keys, TTL, etc.
+ * @param[in] opcode: Message opcode.
+ * @param[in] length: Message length (exclude the message opcode).
+ * @param[in] data: Parameters of Access Payload (exclude the message opcode) to be sent.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_server_model_send_msg(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx, uint32_t opcode,
+ uint16_t length, uint8_t *data);
+
+/**
+ * @brief Send client model message (such as model get, set, etc).
+ *
+ * @param[in] model: BLE Mesh Client Model to which the message belongs.
+ * @param[in] ctx: Message context, includes keys, TTL, etc.
+ * @param[in] opcode: Message opcode.
+ * @param[in] length: Message length (exclude the message opcode).
+ * @param[in] data: Parameters of the Access Payload (exclude the message opcode) to be sent.
+ * @param[in] msg_timeout: Time to get response to the message (in milliseconds).
+ * @param[in] need_rsp: TRUE if the opcode requires the peer device to reply, FALSE otherwise.
+ * @param[in] device_role: Role of the device (Node/Provisioner) that sends the message.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_client_model_send_msg(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx, uint32_t opcode,
+ uint16_t length, uint8_t *data, int32_t msg_timeout,
+ bool need_rsp, esp_ble_mesh_dev_role_t device_role);
+
+/**
+ * @brief Send a model publication message.
+ *
+ * @note Before calling this function, the user needs to ensure that the model
+ * publication message (@ref esp_ble_mesh_model_pub_t.msg) contains a valid
+ * message to be sent. And if users want to update the publishing message,
+ * this API should be called in ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT
+ * with the message updated.
+ *
+ *
+ * @param[in] model: Mesh (client) Model publishing the message.
+ * @param[in] opcode: Message opcode.
+ * @param[in] length: Message length (exclude the message opcode).
+ * @param[in] data: Parameters of the Access Payload (exclude the message opcode) to be sent.
+ * @param[in] device_role: Role of the device (node/provisioner) publishing the message of the type esp_ble_mesh_dev_role_t.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_model_publish(esp_ble_mesh_model_t *model, uint32_t opcode,
+ uint16_t length, uint8_t *data,
+ esp_ble_mesh_dev_role_t device_role);
+
+/**
+ * @brief Reset the provisioning procedure of the local BLE Mesh node.
+ *
+ * @note All provisioning information in this node will be deleted and the node
+ * needs to be reprovisioned. The API function esp_ble_mesh_node_prov_enable()
+ * needs to be called to start a new provisioning procedure.
+ *
+ * @param None.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_node_local_reset(void);
+
+#if defined(CONFIG_BT_MESH_PROVISIONER)
+/**
+ * @brief This function is called to set the node (provisioned device) name.
+ *
+ * @param[in] index: Index of the node in the node queue.
+ * @param[in] name: Name (end by '\0') to be set for the node.
+ *
+ * @note index is obtained from the parameters of ESP_BLE_MESH_PROVISIONER_PROV_COMPLETE_EVT.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_set_node_name(int index, const char *name);
+
+/**
+ * @brief This function is called to get the node (provisioned device) name.
+ *
+ * @param[in] index: Index of the node in the node queue.
+ *
+ * @note index is obtained from the parameters of ESP_BLE_MESH_PROVISIONER_PROV_COMPLETE_EVT.
+ *
+ * @return Node name on success, or NULL on failure.
+ *
+ */
+const char *esp_ble_mesh_provisioner_get_node_name(int index);
+
+/**
+ * @brief This function is called to get the node (provisioned device) index.
+ *
+ * @param[in] name: Name of the node (end by '\0').
+ *
+ * @return Node index on success, or (negative) error code from errno.h on failure.
+ *
+ */
+int esp_ble_mesh_provisioner_get_node_index(const char *name);
+
+/**
+ * @brief This function is called to set the app key for the local BLE Mesh stack.
+ *
+ * @param[in] app_key: The app key to be set for the local BLE Mesh stack.
+ * @param[in] net_idx: The network key index.
+ * @param[in] app_idx: The app key index.
+ *
+ * @note app_key: If set to NULL, app_key will be generated internally.
+ * net_idx: Should be an existing one.
+ * app_idx: If it is going to be generated internally, it should be set to
+ * 0xFFFF, and the new app_idx will be reported via an event.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_add_local_app_key(const uint8_t app_key[16], uint16_t net_idx, uint16_t app_idx);
+
+/**
+ * @brief This function is called by Provisioner to get the local app key value.
+ *
+ * @param[in] net_idx: Network key index.
+ * @param[in] app_idx: Application key index.
+ *
+ * @return App key on success, or NULL on failure.
+ *
+ */
+const uint8_t *esp_ble_mesh_provisioner_get_local_app_key(uint16_t net_idx, uint16_t app_idx);
+
+/**
+ * @brief This function is called by Provisioner to bind own model with proper app key.
+ *
+ * @param[in] element_addr: Provisioner local element address
+ * @param[in] app_idx: Provisioner local appkey index
+ * @param[in] model_id: Provisioner local model id
+ * @param[in] company_id: Provisioner local company id
+ *
+ * @note company_id: If going to bind app_key with local vendor model, company_id
+ * should be set to 0xFFFF.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_bind_app_key_to_local_model(uint16_t element_addr, uint16_t app_idx,
+ uint16_t model_id, uint16_t company_id);
+
+/**
+ * @brief This function is called by Provisioner to add local network key.
+ *
+ * @param[in] net_key: The network key to be added to the Provisioner local BLE Mesh stack.
+ * @param[in] net_idx: The network key index.
+ *
+ * @note net_key: If set to NULL, net_key will be generated internally.
+ * net_idx: If it is going to be generated internally, it should be set to
+ * 0xFFFF, and the new net_idx will be reported via an event.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_add_local_net_key(const uint8_t net_key[16], uint16_t net_idx);
+
+/**
+ * @brief This function is called by Provisioner to get the local network key value.
+ *
+ * @param[in] net_idx: Network key index.
+ *
+ * @return Network key on success, or NULL on failure.
+ *
+ */
+const uint8_t *esp_ble_mesh_provisioner_get_local_net_key(uint16_t net_idx);
+
+#endif /* #if CONFIG_BT_MESH_PROVISIONER */
+
+/**
+ * @brief This function is called to get fast provisioning application key.
+ *
+ * @param[in] net_idx: Network key index.
+ * @param[in] app_idx: Application key index.
+ *
+ * @return Application key on success, or NULL on failure.
+ *
+ */
+const uint8_t *esp_ble_mesh_get_fast_prov_app_key(uint16_t net_idx, uint16_t app_idx);
+
+#endif /* #if CONFIG_BT_MESH */
+#endif /* _ESP_BLE_MESH_NETWORKING_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/core/include/esp_ble_mesh_provisioning_api.h b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_provisioning_api.h
new file mode 100644
index 0000000000..df3bb19357
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_provisioning_api.h
@@ -0,0 +1,324 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_PROVISIONING_API_H_
+#define _ESP_BLE_MESH_PROVISIONING_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+/** @brief: event, event code of provisioning events; param, parameters of provisioning events */
+typedef void (* esp_ble_mesh_prov_cb_t)(esp_ble_mesh_prov_cb_event_t event,
+ esp_ble_mesh_prov_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh provisioning callback.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_prov_callback(esp_ble_mesh_prov_cb_t callback);
+
+/**
+ * @brief Check if a device has been provisioned.
+ *
+ * @param None.
+ *
+ * @return TRUE if the device is provisioned, FALSE if the device is unprovisioned.
+ *
+ */
+bool esp_ble_mesh_node_is_provisioned(void);
+
+/**
+ * @brief Enable specific provisioning bearers to get the device ready for provisioning.
+ *
+ * @note PB-ADV: send unprovisioned device beacon.
+ * PB-GATT: send connectable advertising packets.
+ *
+ * @param bearers: Bit-wise OR of provisioning bearers.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_node_prov_enable(esp_ble_mesh_prov_bearer_t bearers);
+
+/**
+ * @brief Disable specific provisioning bearers to make a device inaccessible for provisioning.
+ *
+ * @param bearers: Bit-wise OR of provisioning bearers.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_node_prov_disable(esp_ble_mesh_prov_bearer_t bearers);
+
+/**
+ * @brief Unprovisioned device set own oob public key & private key pair.
+ *
+ * @param[in] pub_key_x: Unprovisioned device's Public Key X
+ * @param[in] pub_key_y: Unprovisioned device's Public Key Y
+ * @param[in] private_key: Unprovisioned device's Private Key
+ *
+ * @return ESP_OK on success or error code otherwise.
+ */
+esp_err_t esp_ble_mesh_node_set_oob_pub_key(uint8_t pub_key_x[32], uint8_t pub_key_y[32],
+ uint8_t private_key[32]);
+
+/**
+ * @brief Provide provisioning input OOB number.
+ *
+ * @note This is intended to be called if the user has received ESP_BLE_MESH_NODE_PROV_INPUT_EVT
+ * with ESP_BLE_MESH_ENTER_NUMBER as the action.
+ *
+ * @param[in] number: Number input by device.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_node_input_number(uint32_t number);
+
+/**
+ * @brief Provide provisioning input OOB string.
+ *
+ * @note This is intended to be called if the user has received ESP_BLE_MESH_NODE_PROV_INPUT_EVT
+ * with ESP_BLE_MESH_ENTER_STRING as the action.
+ *
+ * @param[in] string: String input by device.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_node_input_string(const char *string);
+
+/**
+ * @brief Using this function, an unprovisioned device can set its own device name,
+ * which will be broadcasted in its advertising data.
+ *
+ * @param[in] name: Unprovisioned device name
+ *
+ * @note This API applicable to PB-GATT mode only by setting the name to the scan response data,
+ * it doesn't apply to PB-ADV mode.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_set_unprovisioned_device_name(const char *name);
+
+#if defined(CONFIG_BT_MESH_PROVISIONER)
+
+/**
+ * @brief Provisioner inputs unprovisioned device's oob public key.
+ *
+ * @param[in] link_idx: The provisioning link index
+ * @param[in] pub_key_x: Unprovisioned device's Public Key X
+ * @param[in] pub_key_y: Unprovisioned device's Public Key Y
+ *
+ * @return ESP_OK on success or error code otherwise.
+ */
+esp_err_t esp_ble_mesh_provisioner_read_oob_pub_key(uint8_t link_idx, uint8_t pub_key_x[32],
+ uint8_t pub_key_y[32]);
+
+/**
+ * @brief Provide provisioning input OOB string.
+ *
+ * This is intended to be called after the esp_ble_mesh_prov_t prov_input_num
+ * callback has been called with ESP_BLE_MESH_ENTER_STRING as the action.
+ *
+ * @param[in] string: String input by Provisioner.
+ * @param[in] link_idx: The provisioning link index.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_input_string(const char *string, uint8_t link_idx);
+
+/**
+ * @brief Provide provisioning input OOB number.
+ *
+ * This is intended to be called after the esp_ble_mesh_prov_t prov_input_num
+ * callback has been called with ESP_BLE_MESH_ENTER_NUMBER as the action.
+ *
+ * @param[in] number: Number input by Provisioner.
+ * @param[in] link_idx: The provisioning link index.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_input_number(uint32_t number, uint8_t link_idx);
+
+/**
+ * @brief Enable one or more provisioning bearers.
+ *
+ * @param[in] bearers: Bit-wise OR of provisioning bearers.
+ *
+ * @note PB-ADV: Enable BLE scan.
+ * PB-GATT: Initialize corresponding BLE Mesh Proxy info.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_prov_enable(esp_ble_mesh_prov_bearer_t bearers);
+
+/**
+ * @brief Disable one or more provisioning bearers.
+ *
+ * @param[in] bearers: Bit-wise OR of provisioning bearers.
+ *
+ * @note PB-ADV: Disable BLE scan.
+ * PB-GATT: Break any existing BLE Mesh Provisioning connections.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_prov_disable(esp_ble_mesh_prov_bearer_t bearers);
+
+/**
+ * @brief Add unprovisioned device info to the unprov_dev queue.
+ *
+ * @param[in] add_dev: Pointer to a struct containing the device information
+ * @param[in] flags: Flags indicate several operations on the device information
+ * - Remove device information from queue after device has been provisioned (BIT0)
+ * - Start provisioning immediately after device is added to queue (BIT1)
+ * - Device can be removed if device queue is full (BIT2)
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ * @note: 1. Currently address type only supports public address and static random address.
+ * 2. If device UUID and/or device address as well as address type already exist in the
+ * device queue, but the bearer is different from the existing one, add operation
+ * will also be successful and it will update the provision bearer supported by
+ * the device.
+ * 3. For example, if the Provisioner wants to add an unprovisioned device info before
+ * receiving its unprovisioned device beacon or Mesh Provisioning advertising packets,
+ * the Provisioner can use this API to add the device info with each one or both of
+ * device UUID and device address added. When the Provisioner gets the device's
+ * advertising packets, it will start provisioning the device internally.
+ * - In this situation, the Provisioner can set bearers with each one or both of
+ * ESP_BLE_MESH_PROV_ADV and ESP_BLE_MESH_PROV_GATT enabled, and cannot set flags
+ * with ADD_DEV_START_PROV_NOW_FLAG enabled.
+ * 4. Another example is when the Provisioner receives the unprovisioned device's beacon or
+ * Mesh Provisioning advertising packets, the advertising packets will be reported on to
+ * the application layer using the callback registered by the function
+ * esp_ble_mesh_register_prov_callback. And in the callback, the Provisioner
+ * can call this API to start provisioning the device.
+ * - If the Provisioner uses PB-ADV to provision, either one or both of device UUID and
+ * device address can be added, bearers shall be set with ESP_BLE_MESH_PROV_ADV
+ * enabled and the flags shall be set with ADD_DEV_START_PROV_NOW_FLAG enabled.
+ * - If the Provisioner uses PB-GATT to provision, both the device UUID and device
+ * address need to be added, bearers shall be set with ESP_BLE_MESH_PROV_GATT enabled,
+ * and the flags shall be set with ADD_DEV_START_PROV_NOW_FLAG enabled.
+ * - If the Provisioner just wants to store the unprovisioned device info when receiving
+ * its advertising packets and start to provision it the next time (e.g. after receiving
+ * its advertising packets again), then it can add the device info with either one or both
+ * of device UUID and device address included. Bearers can be set with either one or both
+ * of ESP_BLE_MESH_PROV_ADV and ESP_BLE_MESH_PROV_GATT enabled (recommend to enable the
+ * bearer which will receive its advertising packets, because if the other bearer is
+ * enabled, the Provisioner is not aware if the device supports the bearer), and flags
+ * cannot be set with ADD_DEV_START_PROV_NOW_FLAG enabled.
+ * - Note: ESP_BLE_MESH_PROV_ADV, ESP_BLE_MESH_PROV_GATT and ADD_DEV_START_PROV_NOW_FLAG
+ * can not be enabled at the same time.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_add_unprov_dev(esp_ble_mesh_unprov_dev_add_t *add_dev,
+ esp_ble_mesh_dev_add_flag_t flags);
+
+/**
+ * @brief Delete device from queue, reset current provisioning link and reset the node.
+ *
+ * @note If the device is in the queue, remove it from the queue; if the device is being
+ * provisioned, terminate the provisioning procedure; if the device has already
+ * been provisioned, reset the device. And either one of the addr or device UUID
+ * can be input.
+ *
+ * @param[in] del_dev: Pointer to a struct containing the device information.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_delete_dev(esp_ble_mesh_device_delete_t *del_dev);
+
+/**
+ * @brief Callback for Provisioner that received advertising packets from unprovisioned devices which are
+ * not in the unprovisioned device queue.
+ *
+ * Report on the unprovisioned device beacon and mesh provisioning service adv data to application.
+ *
+ * @param[in] addr: Pointer to the unprovisioned device address.
+ * @param[in] addr_type: Unprovisioned device address type.
+ * @param[in] adv_type: Adv packet type(ADV_IND or ADV_NONCONN_IND).
+ * @param[in] dev_uuid: Unprovisioned device UUID pointer.
+ * @param[in] oob_info: OOB information of the unprovisioned device.
+ * @param[in] bearer: Adv packet received from PB-GATT or PB-ADV bearer.
+ *
+ */
+typedef void (*esp_ble_mesh_prov_adv_cb_t)(const esp_bd_addr_t addr, const esp_ble_addr_type_t addr_type,
+ const uint8_t adv_type, const uint8_t *dev_uuid,
+ uint16_t oob_info, esp_ble_mesh_prov_bearer_t bearer);
+
+/**
+ * @brief This function is called by Provisioner to set the part of the device UUID
+ * to be compared before starting to provision.
+ *
+ * @param[in] match_val: Value to be compared with the part of the device UUID.
+ * @param[in] match_len: Length of the compared match value.
+ * @param[in] offset: Offset of the device UUID to be compared (based on zero).
+ * @param[in] prov_after_match: Flag used to indicate whether provisioner should start to provision
+ * the device immediately if the part of the UUID matches.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_set_dev_uuid_match(const uint8_t *match_val, uint8_t match_len,
+ uint8_t offset, bool prov_after_match);
+
+/**
+ * @brief This function is called by Provisioner to set provisioning data information
+ * before starting to provision.
+ *
+ * @param[in] prov_data_info: Pointer to a struct containing net_idx or flags or iv_index.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_provisioner_set_prov_data_info(esp_ble_mesh_prov_data_info_t *prov_data_info);
+
+#endif /* #if CONFIG_BT_MESH_PROVISIONER */
+
+/**
+ * @brief This function is called to set provisioning data information before starting
+ * fast provisioning.
+ *
+ * @param[in] fast_prov_info: Pointer to a struct containing unicast address range, net_idx, etc.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_set_fast_prov_info(esp_ble_mesh_fast_prov_info_t *fast_prov_info);
+
+/**
+ * @brief This function is called to start/suspend/exit fast provisioning.
+ *
+ * @param[in] action: fast provisioning action (i.e. enter, suspend, exit).
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_set_fast_prov_action(esp_ble_mesh_fast_prov_action_t action);
+
+#endif /* #if CONFIG_BT_MESH */
+#endif /* _ESP_BLE_MESH_PROVISIONING_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/core/include/esp_ble_mesh_proxy_api.h b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_proxy_api.h
new file mode 100644
index 0000000000..67a785dda6
--- /dev/null
+++ b/components/bt/ble_mesh/api/core/include/esp_ble_mesh_proxy_api.h
@@ -0,0 +1,59 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_PROXY_API_H_
+#define _ESP_BLE_MESH_PROXY_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+/**
+ * @brief Enable advertising with Node Identity.
+ *
+ * @note This API requires that GATT Proxy support be enabled. Once called,
+ * each subnet starts advertising using Node Identity for the next 60
+ * seconds, and after 60s Network ID will be advertised.
+ * Under normal conditions, the BLE Mesh Proxy Node Identity and
+ * Network ID advertising will be enabled automatically by BLE Mesh
+ * stack after the device is provisioned.
+ *
+ * @param None.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_proxy_identity_enable(void);
+
+/**
+ * @brief Enable BLE Mesh GATT Proxy Service.
+ *
+ * @param None.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_proxy_gatt_enable(void);
+
+/**
+ * @brief Disconnect the BLE Mesh GATT Proxy connection if there is any, and
+ * disable the BLE Mesh GATT Proxy Service.
+ *
+ * @param None.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_proxy_gatt_disable(void);
+
+#endif /* _ESP_BLE_MESH_PROXY_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/esp_ble_mesh_defs.h b/components/bt/ble_mesh/api/esp_ble_mesh_defs.h
new file mode 100644
index 0000000000..05e7c1b933
--- /dev/null
+++ b/components/bt/ble_mesh/api/esp_ble_mesh_defs.h
@@ -0,0 +1,1480 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_DEFS_H_
+#define _ESP_BLE_MESH_DEFS_H_
+
+#include
+
+#include "esp_bt_defs.h"
+
+#include "mesh.h"
+#include "proxy.h"
+#include "include/proxy.h"
+#include "mesh_access.h"
+#include "mesh_main.h"
+#include "foundation.h"
+#include "model_op.h"
+#include "common.h"
+
+#if CONFIG_BT_MESH
+
+#ifndef MIN
+#define MIN(x, y) (((x) < (y)) ? (x) : (y))
+#endif /* #ifndef MIN */
+
+#define ESP_BLE_MESH_SDU_MAX_LEN 384
+
+/*!< The maximum length of a BLE Mesh provisioned node name */
+#define ESP_BLE_MESH_NODE_NAME_MAX_LEN 31
+
+/*!< The maximum length of a BLE Mesh unprovisioned device name */
+#define ESP_BLE_MESH_DEVICE_NAME_MAX_LEN DEVICE_NAME_SIZE
+
+/*!< Define the BLE Mesh octet 16 bytes size */
+#define ESP_BLE_MESH_OCTET16_LEN 16
+typedef uint8_t esp_ble_mesh_octet16_t[ESP_BLE_MESH_OCTET16_LEN];
+
+/*!< Define the BLE Mesh octet 8 bytes size */
+#define ESP_BLE_MESH_OCTET8_LEN 8
+typedef uint8_t esp_ble_mesh_octet8_t[ESP_BLE_MESH_OCTET8_LEN];
+
+#define ESP_BLE_MESH_ADDR_UNASSIGNED BT_MESH_ADDR_UNASSIGNED
+#define ESP_BLE_MESH_ADDR_ALL_NODES BT_MESH_ADDR_ALL_NODES
+#define ESP_BLE_MESH_ADDR_PROXIES BT_MESH_ADDR_PROXIES
+#define ESP_BLE_MESH_ADDR_FRIENDS BT_MESH_ADDR_FRIENDS
+#define ESP_BLE_MESH_ADDR_RELAYS BT_MESH_ADDR_RELAYS
+
+#define ESP_BLE_MESH_KEY_UNUSED BT_MESH_KEY_UNUSED
+#define ESP_BLE_MESH_KEY_DEV BT_MESH_KEY_DEV
+
+#define ESP_BLE_MESH_KEY_PRIMARY BT_MESH_KEY_PRIMARY
+#define ESP_BLE_MESH_KEY_ANY BT_MESH_KEY_ANY
+
+/*!< Primary Network Key index */
+#define ESP_BLE_MESH_NET_PRIMARY BT_MESH_NET_PRIMARY
+
+/*!< Relay state value */
+#define ESP_BLE_MESH_RELAY_DISABLED BT_MESH_RELAY_DISABLED
+#define ESP_BLE_MESH_RELAY_ENABLED BT_MESH_RELAY_ENABLED
+#define ESP_BLE_MESH_RELAY_NOT_SUPPORTED BT_MESH_RELAY_NOT_SUPPORTED
+
+/*!< Beacon state value */
+#define ESP_BLE_MESH_BEACON_DISABLED BT_MESH_BEACON_DISABLED
+#define ESP_BLE_MESH_BEACON_ENABLED BT_MESH_BEACON_ENABLED
+
+/*!< GATT Proxy state value */
+#define ESP_BLE_MESH_GATT_PROXY_DISABLED BT_MESH_GATT_PROXY_DISABLED
+#define ESP_BLE_MESH_GATT_PROXY_ENABLED BT_MESH_GATT_PROXY_ENABLED
+#define ESP_BLE_MESH_GATT_PROXY_NOT_SUPPORTED BT_MESH_GATT_PROXY_NOT_SUPPORTED
+
+/*!< Friend state value */
+#define ESP_BLE_MESH_FRIEND_DISABLED BT_MESH_FRIEND_DISABLED
+#define ESP_BLE_MESH_FRIEND_ENABLED BT_MESH_FRIEND_ENABLED
+#define ESP_BLE_MESH_FRIEND_NOT_SUPPORTED BT_MESH_FRIEND_NOT_SUPPORTED
+
+/*!< Node identity state value */
+#define ESP_BLE_MESH_NODE_IDENTITY_STOPPED BT_MESH_NODE_IDENTITY_STOPPED
+#define ESP_BLE_MESH_NODE_IDENTITY_RUNNING BT_MESH_NODE_IDENTITY_RUNNING
+#define ESP_BLE_MESH_NODE_IDENTITY_NOT_SUPPORTED BT_MESH_NODE_IDENTITY_NOT_SUPPORTED
+
+/*!< Supported features */
+#define ESP_BLE_MESH_FEATURE_RELAY BT_MESH_FEAT_RELAY
+#define ESP_BLE_MESH_FEATURE_PROXY BT_MESH_FEAT_PROXY
+#define ESP_BLE_MESH_FEATURE_FRIEND BT_MESH_FEAT_FRIEND
+#define ESP_BLE_MESH_FEATURE_LOW_POWER BT_MESH_FEAT_LOW_POWER
+#define ESP_BLE_MESH_FEATURE_ALL_SUPPORTED BT_MESH_FEAT_SUPPORTED
+
+#define ESP_BLE_MESH_ADDR_IS_UNICAST(addr) BT_MESH_ADDR_IS_UNICAST(addr)
+#define ESP_BLE_MESH_ADDR_IS_GROUP(addr) BT_MESH_ADDR_IS_GROUP(addr)
+#define ESP_BLE_MESH_ADDR_IS_VIRTUAL(addr) BT_MESH_ADDR_IS_VIRTUAL(addr)
+#define ESP_BLE_MESH_ADDR_IS_RFU(addr) BT_MESH_ADDR_IS_RFU(addr)
+
+#define ESP_BLE_MESH_INVALID_NODE_INDEX (-1)
+
+/*!< Foundation Models */
+#define ESP_BLE_MESH_MODEL_ID_CONFIG_SRV BT_MESH_MODEL_ID_CFG_SRV
+#define ESP_BLE_MESH_MODEL_ID_CONFIG_CLI BT_MESH_MODEL_ID_CFG_CLI
+#define ESP_BLE_MESH_MODEL_ID_HEALTH_SRV BT_MESH_MODEL_ID_HEALTH_SRV
+#define ESP_BLE_MESH_MODEL_ID_HEALTH_CLI BT_MESH_MODEL_ID_HEALTH_CLI
+
+/*!< Models from the Mesh Model Specification */
+#define ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV BT_MESH_MODEL_ID_GEN_ONOFF_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_CLI BT_MESH_MODEL_ID_GEN_ONOFF_CLI
+#define ESP_BLE_MESH_MODEL_ID_GEN_LEVEL_SRV BT_MESH_MODEL_ID_GEN_LEVEL_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_LEVEL_CLI BT_MESH_MODEL_ID_GEN_LEVEL_CLI
+#define ESP_BLE_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_SRV BT_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_CLI BT_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_CLI
+#define ESP_BLE_MESH_MODEL_ID_GEN_POWER_ONOFF_SRV BT_MESH_MODEL_ID_GEN_POWER_ONOFF_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_POWER_ONOFF_SETUP_SRV BT_MESH_MODEL_ID_GEN_POWER_ONOFF_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_POWER_ONOFF_CLI BT_MESH_MODEL_ID_GEN_POWER_ONOFF_CLI
+#define ESP_BLE_MESH_MODEL_ID_GEN_POWER_LEVEL_SRV BT_MESH_MODEL_ID_GEN_POWER_LEVEL_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_POWER_LEVEL_SETUP_SRV BT_MESH_MODEL_ID_GEN_POWER_LEVEL_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_POWER_LEVEL_CLI BT_MESH_MODEL_ID_GEN_POWER_LEVEL_CLI
+#define ESP_BLE_MESH_MODEL_ID_GEN_BATTERY_SRV BT_MESH_MODEL_ID_GEN_BATTERY_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_BATTERY_CLI BT_MESH_MODEL_ID_GEN_BATTERY_CLI
+#define ESP_BLE_MESH_MODEL_ID_GEN_LOCATION_SRV BT_MESH_MODEL_ID_GEN_LOCATION_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_LOCATION_SETUP_SRV BT_MESH_MODEL_ID_GEN_LOCATION_SETUPSRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_LOCATION_CLI BT_MESH_MODEL_ID_GEN_LOCATION_CLI
+#define ESP_BLE_MESH_MODEL_ID_GEN_ADMIN_PROP_SRV BT_MESH_MODEL_ID_GEN_ADMIN_PROP_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_MANUFACTURER_PROP_SRV BT_MESH_MODEL_ID_GEN_MANUFACTURER_PROP_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_USER_PROP_SRV BT_MESH_MODEL_ID_GEN_USER_PROP_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_CLIENT_PROP_SRV BT_MESH_MODEL_ID_GEN_CLIENT_PROP_SRV
+#define ESP_BLE_MESH_MODEL_ID_GEN_PROP_CLI BT_MESH_MODEL_ID_GEN_PROP_CLI
+#define ESP_BLE_MESH_MODEL_ID_SENSOR_SRV BT_MESH_MODEL_ID_SENSOR_SRV
+#define ESP_BLE_MESH_MODEL_ID_SENSOR_SETUP_SRV BT_MESH_MODEL_ID_SENSOR_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_SENSOR_CLI BT_MESH_MODEL_ID_SENSOR_CLI
+#define ESP_BLE_MESH_MODEL_ID_TIME_SRV BT_MESH_MODEL_ID_TIME_SRV
+#define ESP_BLE_MESH_MODEL_ID_TIME_SETUP_SRV BT_MESH_MODEL_ID_TIME_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_TIME_CLI BT_MESH_MODEL_ID_TIME_CLI
+#define ESP_BLE_MESH_MODEL_ID_SCENE_SRV BT_MESH_MODEL_ID_SCENE_SRV
+#define ESP_BLE_MESH_MODEL_ID_SCENE_SETUP_SRV BT_MESH_MODEL_ID_SCENE_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_SCENE_CLI BT_MESH_MODEL_ID_SCENE_CLI
+#define ESP_BLE_MESH_MODEL_ID_SCHEDULER_SRV BT_MESH_MODEL_ID_SCHEDULER_SRV
+#define ESP_BLE_MESH_MODEL_ID_SCHEDULER_SETUP_SRV BT_MESH_MODEL_ID_SCHEDULER_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_SCHEDULER_CLI BT_MESH_MODEL_ID_SCHEDULER_CLI
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_CLI BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_CLI
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_CTL_SRV BT_MESH_MODEL_ID_LIGHT_CTL_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV BT_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_CTL_CLI BT_MESH_MODEL_ID_LIGHT_CTL_CLI
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_CTL_TEMP_SRV BT_MESH_MODEL_ID_LIGHT_CTL_TEMP_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_HSL_SRV BT_MESH_MODEL_ID_LIGHT_HSL_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV BT_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_HSL_CLI BT_MESH_MODEL_ID_LIGHT_HSL_CLI
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_HSL_HUE_SRV BT_MESH_MODEL_ID_LIGHT_HSL_HUE_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_HSL_SAT_SRV BT_MESH_MODEL_ID_LIGHT_HSL_SAT_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_XYL_SRV BT_MESH_MODEL_ID_LIGHT_XYL_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV BT_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_XYL_CLI BT_MESH_MODEL_ID_LIGHT_XYL_CLI
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_LC_SRV BT_MESH_MODEL_ID_LIGHT_LC_SRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_LC_SETUP_SRV BT_MESH_MODEL_ID_LIGHT_LC_SETUPSRV
+#define ESP_BLE_MESH_MODEL_ID_LIGHT_LC_CLI BT_MESH_MODEL_ID_LIGHT_LC_CLI
+
+/*!< The following opcodes will only be used in the esp_ble_mesh_config_client_get_state function. */
+typedef uint32_t esp_ble_mesh_opcode_config_client_get_t; /*!< esp_ble_mesh_opcode_config_client_get_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by esp_ble_mesh_config_client_get_state */
+#define ESP_BLE_MESH_MODEL_OP_BEACON_GET OP_BEACON_GET /*!< To determine the Secure Network Beacon state of a Configuration Server */
+#define ESP_BLE_MESH_MODEL_OP_COMPOSITION_DATA_GET OP_DEV_COMP_DATA_GET /*!< To determine the Composition Data state of a Configuration Server, a Configuration
+ Client shall send a Config Composition Data Get message with the Page field value set
+ to 0xFF. The response is a Config Composition Data Status message that contains the last
+ page of the Composition Data state. If the Page field of the Config Composition Data Status
+ message contains a non-zero value, then the Configuration Client shall send another Composition
+ Data Get message with the Page field value set to one less than the Page field value of the
+ Config Composition Data Status message. */
+#define ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_GET OP_DEFAULT_TTL_GET /*!< To determine the Default TTL state of a Configuration Server */
+#define ESP_BLE_MESH_MODEL_OP_GATT_PROXY_GET OP_GATT_PROXY_GET /*!< To determine the GATT Proxy state of a Configuration Server */
+#define ESP_BLE_MESH_MODEL_OP_RELAY_GET OP_RELAY_GET /*!< To determine the Relay and Relay Retransmit states of a Configuration Server */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_PUB_GET OP_MOD_PUB_GET /*!< To determine the Publish Address, Publish AppKey Index, CredentialFlag,
+ Publish Period, Publish Retransmit Count, Publish Retransmit Interval Steps,
+ and Publish TTL states of a particular Model within the element */
+#define ESP_BLE_MESH_MODEL_OP_FRIEND_GET OP_FRIEND_GET /*!< To determine the Friend state of a Configuration Server */
+#define ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_GET OP_HEARTBEAT_PUB_GET /*!< To determine the Heartbeat Subscription Source, Heartbeat Subscription Destination,
+ Heartbeat Subscription Count Log, Heartbeat Subscription Period Log, Heartbeat
+ Subscription Min Hops, and Heartbeat Subscription Max Hops states of a node */
+#define ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_GET OP_HEARTBEAT_SUB_GET /*!< To determine the Heartbeat Subscription Source, Heartbeat Subscription Destination,
+ Heartbeat Subscription Count Log, Heartbeat Subscription Period Log, Heartbeat
+ Subscription Min Hops, and Heartbeat Subscription Max Hops states of a node */
+#define ESP_BLE_MESH_MODEL_OP_NET_KEY_GET OP_NET_KEY_GET /*!< To determine all NetKeys known to the node */
+#define ESP_BLE_MESH_MODEL_OP_APP_KEY_GET OP_APP_KEY_GET /*!< To determine all AppKeys bound to the NetKey */
+#define ESP_BLE_MESH_MODEL_OP_NODE_IDENTITY_GET OP_NODE_IDENTITY_GET /*!< To get the current Node Identity state for a subnet */
+#define ESP_BLE_MESH_MODEL_OP_SIG_MODEL_SUB_GET OP_MOD_SUB_GET /*!< To get the list of subscription addresses of a model within the element */
+#define ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_SUB_GET OP_MOD_SUB_GET_VND /*!< To get the list of subscription addresses of a model within the element */
+#define ESP_BLE_MESH_MODEL_OP_SIG_MODEL_APP_GET OP_SIG_MOD_APP_GET /*!< To request report of all AppKeys bound to the SIG Model */
+#define ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_APP_GET OP_VND_MOD_APP_GET /*!< To request report of all AppKeys bound to the Vendor Model */
+#define ESP_BLE_MESH_MODEL_OP_KEY_REFRESH_PHASE_GET OP_KRP_GET /*!< To get the current Key Refresh Phase state of the identified network key */
+#define ESP_BLE_MESH_MODEL_OP_LPN_POLLTIMEOUT_GET OP_LPN_TIMEOUT_GET /*!< To get the current value of PollTimeout timer of the Low Power node within a Friend node */
+#define ESP_BLE_MESH_MODEL_OP_NETWORK_TRANSMIT_GET OP_NET_TRANSMIT_GET /*!< To get the current Network Transmit state of a node */
+
+/*!< The following opcodes will only be used in the esp_ble_mesh_config_client_set_state function. */
+typedef uint32_t esp_ble_mesh_opcode_config_client_set_t; /*!< esp_ble_mesh_opcode_config_client_set_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by esp_ble_mesh_config_client_set_state */
+#define ESP_BLE_MESH_MODEL_OP_BEACON_SET OP_BEACON_SET /*!< Set the Secure Network Beacon state of a Configuration Server with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_SET OP_DEFAULT_TTL_SET /*!< Set the Default TTL state of a Configuration Server with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_GATT_PROXY_SET OP_GATT_PROXY_SET /*!< Determine the GATT Proxy state of a Configuration Server */
+#define ESP_BLE_MESH_MODEL_OP_RELAY_SET OP_RELAY_SET /*!< Set the Relay and Relay Retransmit states of a Configuration Server with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_PUB_SET OP_MOD_PUB_SET /*!< Set the Publish Address, Publish AppKey Index, CredentialFlag, Publish
+ Period, Publish Retransmit Count, Publish Retransmit Interval Steps, and
+ Publish TTL states of a particular model within the element with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_ADD OP_MOD_SUB_ADD /*!< Add the address to the Subscription List state of a particular model
+ within the element with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_ADD OP_MOD_SUB_VA_ADD /*!< Add the Label UUID to the Subscription List state of a particular model
+ within the element with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE OP_MOD_SUB_DEL /*!< Delete the address from the Subscription List state of a particular
+ model within the element with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_DELETE OP_MOD_SUB_VA_DEL /*!< Delete the Label UUID from the Subscription List state of a particular
+ model within the element with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_OVERWRITE OP_MOD_SUB_OVERWRITE /*!< Clear the Subscription List and add the address to the Subscription List
+ state of a particular Model within the element with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_OVERWRITE OP_MOD_SUB_VA_OVERWRITE /*!< Clear the Subscription List and add the Label UUID to the Subscription
+ List state of a particular model within the element with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_NET_KEY_ADD OP_NET_KEY_ADD /*!< Add the NetKey identified by NetKeyIndex to the NetKey List state with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_APP_KEY_ADD OP_APP_KEY_ADD /*!< Add the AppKey to the AppKey List and bind it to the NetKey identified
+ by the NetKeyIndex of a Configuration Server with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_APP_BIND OP_MOD_APP_BIND /*!< Bind the AppKey to a model of a particular element of a Configuration Server with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_NODE_RESET OP_NODE_RESET /*!< Reset a node (other than a Provisioner) and remove it from the network */
+#define ESP_BLE_MESH_MODEL_OP_FRIEND_SET OP_FRIEND_SET /*!< Set the Friend state of a Configuration Server with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_SET OP_HEARTBEAT_PUB_SET /*!< Set the Heartbeat Publication Destination, Heartbeat Publication Count,
+ Heartbeat Publication Period, Heartbeat Publication TTL, Publication Features,
+ and Publication NetKey Index of a node with acknowledgment */
+#define ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_SET OP_HEARTBEAT_SUB_SET /*!< Determine the Heartbeat Subscription Source, Heartbeat Subscription Destination,
+ Heartbeat Subscription Count Log, Heartbeat Subscription Period Log, Heartbeat
+ Subscription Min Hops, and Heartbeat Subscription Max Hops states of a node */
+#define ESP_BLE_MESH_MODEL_OP_NET_KEY_UPDATE OP_NET_KEY_UPDATE /*!< To update a NetKey on a node */
+#define ESP_BLE_MESH_MODEL_OP_NET_KEY_DELETE OP_NET_KEY_DEL /*!< To delete a NetKey on a NetKey List from a node */
+#define ESP_BLE_MESH_MODEL_OP_APP_KEY_UPDATE OP_APP_KEY_UPDATE /*!< To update an AppKey value on the AppKey List on a node */
+#define ESP_BLE_MESH_MODEL_OP_APP_KEY_DELETE OP_APP_KEY_DEL /*!< To delete an AppKey from the AppKey List on a node */
+#define ESP_BLE_MESH_MODEL_OP_NODE_IDENTITY_SET OP_NODE_IDENTITY_SET /*!< To set the current Node Identity state for a subnet */
+#define ESP_BLE_MESH_MODEL_OP_KEY_REFRESH_PHASE_SET OP_KRP_SET /*!< To set the Key Refresh Phase state of the identified network key */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_PUB_VIRTUAL_ADDR_SET OP_MOD_PUB_VA_SET /*!< To set the model Publication state of an outgoing message that originates from a model */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE_ALL OP_MOD_SUB_DEL_ALL /*!< To discard the Subscription List of a model */
+#define ESP_BLE_MESH_MODEL_OP_MODEL_APP_UNBIND OP_MOD_APP_UNBIND /*!< To remove the binding between an AppKey and a model */
+#define ESP_BLE_MESH_MODEL_OP_NETWORK_TRANSMIT_SET OP_NET_TRANSMIT_SET /*!< To set the Network Transmit state of a node */
+
+/*!< The following opcodes are used by the BLE Mesh Config Server Model internally to respond to the Config Client Model's request messages */
+typedef uint32_t esp_ble_mesh_config_model_status_t; /*!< esp_ble_mesh_config_model_status_t belongs to esp_ble_mesh_opcode_t, this typedef
+ is only used to locate the opcodes used by the Config Model messages */
+#define ESP_BLE_MESH_MODEL_OP_BEACON_STATUS OP_BEACON_STATUS
+#define ESP_BLE_MESH_MODEL_OP_COMPOSITION_DATA_STATUS OP_DEV_COMP_DATA_STATUS
+#define ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_STATUS OP_DEFAULT_TTL_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GATT_PROXY_STATUS OP_GATT_PROXY_STATUS
+#define ESP_BLE_MESH_MODEL_OP_RELAY_STATUS OP_RELAY_STATUS
+#define ESP_BLE_MESH_MODEL_OP_MODEL_PUB_STATUS OP_MOD_PUB_STATUS
+#define ESP_BLE_MESH_MODEL_OP_MODEL_SUB_STATUS OP_MOD_SUB_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SIG_MODEL_SUB_LIST OP_MOD_SUB_LIST
+#define ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_SUB_LIST OP_MOD_SUB_LIST_VND
+#define ESP_BLE_MESH_MODEL_OP_NET_KEY_STATUS OP_NET_KEY_STATUS
+#define ESP_BLE_MESH_MODEL_OP_NET_KEY_LIST OP_NET_KEY_LIST
+#define ESP_BLE_MESH_MODEL_OP_APP_KEY_STATUS OP_APP_KEY_STATUS
+#define ESP_BLE_MESH_MODEL_OP_APP_KEY_LIST OP_APP_KEY_LIST
+#define ESP_BLE_MESH_MODEL_OP_NODE_IDENTITY_STATUS OP_NODE_IDENTITY_STATUS
+#define ESP_BLE_MESH_MODEL_OP_MODEL_APP_STATUS OP_MOD_APP_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SIG_MODEL_APP_LIST OP_SIG_MOD_APP_LIST
+#define ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_APP_LIST OP_VND_MOD_APP_LIST
+#define ESP_BLE_MESH_MODEL_OP_NODE_RESET_STATUS OP_NODE_RESET_STATUS
+#define ESP_BLE_MESH_MODEL_OP_FRIEND_STATUS OP_FRIEND_STATUS
+#define ESP_BLE_MESH_MODEL_OP_KEY_REFRESH_PHASE_STATUS OP_KRP_STATUS
+#define ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_STATUS OP_HEARTBEAT_PUB_STATUS
+#define ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_STATUS OP_HEARTBEAT_SUB_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LPN_POLLTIMEOUT_STATUS OP_LPN_TIMEOUT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_NETWORK_TRANSMIT_STATUS OP_NET_TRANSMIT_STATUS
+
+/*!< The following opcodes will only be used in the esp_ble_mesh_health_client_get_state function. */
+typedef uint32_t esp_ble_mesh_opcode_health_client_get_t; /*!< esp_ble_mesh_opcode_health_client_get_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by esp_ble_mesh_health_client_get_state */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_GET OP_HEALTH_FAULT_GET /*!< Get the current Registered Fault state */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_GET OP_HEALTH_PERIOD_GET /*!< Get the current Health Period state */
+#define ESP_BLE_MESH_MODEL_OP_ATTENTION_GET OP_ATTENTION_GET /*!< Get the current Attention Timer state */
+
+/*!< The following opcodes will only be used in the esp_ble_mesh_health_client_set_state function. */
+typedef uint32_t esp_ble_mesh_opcode_health_client_set_t; /*!< esp_ble_mesh_opcode_health_client_set_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by esp_ble_mesh_health_client_set_state */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR OP_HEALTH_FAULT_CLEAR /*!< Clear Health Fault acknowledged */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR_UNACK OP_HEALTH_FAULT_CLEAR_UNREL /*!< Clear Health Fault Unacknowledged */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST OP_HEALTH_FAULT_TEST /*!< Invoke Health Fault Test acknowledged */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST_UNACK OP_HEALTH_FAULT_TEST_UNREL /*!< Invoke Health Fault Test unacknowledged */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET OP_HEALTH_PERIOD_SET /*!< Set Health Period acknowledged */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET_UNACK OP_HEALTH_PERIOD_SET_UNREL /*!< Set Health Period unacknowledged */
+#define ESP_BLE_MESH_MODEL_OP_ATTENTION_SET OP_ATTENTION_SET /*!< Set Health Attention acknowledged of the Health Server */
+#define ESP_BLE_MESH_MODEL_OP_ATTENTION_SET_UNACK OP_ATTENTION_SET_UNREL /*!< Set Health Attention Unacknowledged of the Health Server */
+
+/*!< The following opcodes are used by the BLE Mesh Health Server Model internally to respond to the Health Client Model's request messages */
+typedef uint32_t esp_ble_mesh_health_model_status_t; /*!< esp_ble_mesh_health_model_status_t belongs to esp_ble_mesh_opcode_t, this typedef
+ is only used to locate the opcodes used by the Health Model messages */
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_CURRENT_STATUS OP_HEALTH_CURRENT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_STATUS OP_HEALTH_FAULT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_STATUS OP_HEALTH_PERIOD_STATUS
+#define ESP_BLE_MESH_MODEL_OP_ATTENTION_STATUS OP_ATTENTION_STATUS
+
+typedef uint32_t esp_ble_mesh_generic_message_opcode_t; /*!< esp_ble_mesh_generic_message_opcode_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by functions
+ esp_ble_mesh_generic_client_get_state & esp_ble_mesh_generic_client_set_state */
+/*!< Generic OnOff Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_GET BT_MESH_MODEL_OP_GEN_ONOFF_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET BT_MESH_MODEL_OP_GEN_ONOFF_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK BT_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS BT_MESH_MODEL_OP_GEN_ONOFF_STATUS
+
+/*!< Generic Level Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_LEVEL_GET BT_MESH_MODEL_OP_GEN_LEVEL_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_LEVEL_SET BT_MESH_MODEL_OP_GEN_LEVEL_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_LEVEL_SET_UNACK BT_MESH_MODEL_OP_GEN_LEVEL_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_LEVEL_STATUS BT_MESH_MODEL_OP_GEN_LEVEL_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_DELTA_SET BT_MESH_MODEL_OP_GEN_DELTA_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_DELTA_SET_UNACK BT_MESH_MODEL_OP_GEN_DELTA_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_MOVE_SET BT_MESH_MODEL_OP_GEN_MOVE_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_MOVE_SET_UNACK BT_MESH_MODEL_OP_GEN_MOVE_SET_UNACK
+
+/*!< Generic Default Transition Time Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_GET BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET_UNACK BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS
+
+/*!< Generic Power OnOff Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONPOWERUP_GET BT_MESH_MODEL_OP_GEN_ONPOWERUP_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS BT_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS
+
+/*!< Generic Power OnOff Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONPOWERUP_SET BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_ONPOWERUP_SET_UNACK BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET_UNACK
+
+/*!< Generic Power Level Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_LEVEL_GET BT_MESH_MODEL_OP_GEN_POWER_LEVEL_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_LEVEL_SET BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_LEVEL_SET_UNACK BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS BT_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_LAST_GET BT_MESH_MODEL_OP_GEN_POWER_LAST_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_LAST_STATUS BT_MESH_MODEL_OP_GEN_POWER_LAST_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_DEFAULT_GET BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_RANGE_GET BT_MESH_MODEL_OP_GEN_POWER_RANGE_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS BT_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS
+
+/*!< Generic Power Level Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_RANGE_SET BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_POWER_RANGE_SET_UNACK BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET_UNACK
+
+/*!< Generic Battery Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_BATTERY_GET BT_MESH_MODEL_OP_GEN_BATTERY_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_BATTERY_STATUS BT_MESH_MODEL_OP_GEN_BATTERY_STATUS
+
+/*!< Generic Location Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_GLOBAL_GET BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_LOCAL_GET BT_MESH_MODEL_OP_GEN_LOC_LOCAL_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS BT_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS
+
+/*!< Generic Location Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET_UNACK BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_LOCAL_SET BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_LOC_LOCAL_SET_UNACK BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET_UNACK
+
+/*!< Generic Manufacturer Property Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTIES_GET BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTIES_STATUS BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_GET BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_SET BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_STATUS BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_STATUS
+
+/*!< Generic Admin Property Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS
+
+/*!< Generic User Property Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS
+#define ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_GET BT_MESH_MODEL_OP_GEN_USER_PROPERTY_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET
+#define ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS BT_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS
+
+/*!< Generic Client Property Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET
+#define ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS
+
+typedef uint32_t esp_ble_mesh_sensor_message_opcode_t; /*!< esp_ble_mesh_sensor_message_opcode_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by functions
+ esp_ble_mesh_sensor_client_get_state & esp_ble_mesh_sensor_client_set_state */
+/*!< Sensor Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_GET BT_MESH_MODEL_OP_SENSOR_GET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_STATUS BT_MESH_MODEL_OP_SENSOR_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_GET BT_MESH_MODEL_OP_SENSOR_COLUMN_GET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_STATUS BT_MESH_MODEL_OP_SENSOR_COLUMN_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_GET BT_MESH_MODEL_OP_SENSOR_SERIES_GET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_STATUS BT_MESH_MODEL_OP_SENSOR_SERIES_STATUS
+
+/*!< Sensor Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_GET BT_MESH_MODEL_OP_SENSOR_CADENCE_GET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET BT_MESH_MODEL_OP_SENSOR_CADENCE_SET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET_UNACK BT_MESH_MODEL_OP_SENSOR_CADENCE_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_STATUS BT_MESH_MODEL_OP_SENSOR_CADENCE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_GET BT_MESH_MODEL_OP_SENSOR_SETTINGS_GET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS BT_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_GET BT_MESH_MODEL_OP_SENSOR_SETTING_GET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET BT_MESH_MODEL_OP_SENSOR_SETTING_SET
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET_UNACK BT_MESH_MODEL_OP_SENSOR_SETTING_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_STATUS BT_MESH_MODEL_OP_SENSOR_SETTING_STATUS
+
+typedef uint32_t esp_ble_mesh_time_scene_message_opcode_t; /*!< esp_ble_mesh_time_scene_message_opcode_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by functions
+ esp_ble_mesh_time_scene_client_get_state & esp_ble_mesh_time_scene_client_set_state */
+/*!< Time Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_TIME_GET BT_MESH_MODEL_OP_TIME_GET
+#define ESP_BLE_MESH_MODEL_OP_TIME_SET BT_MESH_MODEL_OP_TIME_SET
+#define ESP_BLE_MESH_MODEL_OP_TIME_STATUS BT_MESH_MODEL_OP_TIME_STATUS
+#define ESP_BLE_MESH_MODEL_OP_TIME_ROLE_GET BT_MESH_MODEL_OP_TIME_ROLE_GET
+#define ESP_BLE_MESH_MODEL_OP_TIME_ROLE_SET BT_MESH_MODEL_OP_TIME_ROLE_SET
+#define ESP_BLE_MESH_MODEL_OP_TIME_ROLE_STATUS BT_MESH_MODEL_OP_TIME_ROLE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_TIME_ZONE_GET BT_MESH_MODEL_OP_TIME_ZONE_GET
+#define ESP_BLE_MESH_MODEL_OP_TIME_ZONE_SET BT_MESH_MODEL_OP_TIME_ZONE_SET
+#define ESP_BLE_MESH_MODEL_OP_TIME_ZONE_STATUS BT_MESH_MODEL_OP_TIME_ZONE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_TAI_UTC_DELTA_GET BT_MESH_MODEL_OP_TAI_UTC_DELTA_GET
+#define ESP_BLE_MESH_MODEL_OP_TAI_UTC_DELTA_SET BT_MESH_MODEL_OP_TAI_UTC_DELTA_SET
+#define ESP_BLE_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS BT_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS
+
+/*!< Scene Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_SCENE_GET BT_MESH_MODEL_OP_SCENE_GET
+#define ESP_BLE_MESH_MODEL_OP_SCENE_RECALL BT_MESH_MODEL_OP_SCENE_RECALL
+#define ESP_BLE_MESH_MODEL_OP_SCENE_RECALL_UNACK BT_MESH_MODEL_OP_SCENE_RECALL_UNACK
+#define ESP_BLE_MESH_MODEL_OP_SCENE_STATUS BT_MESH_MODEL_OP_SCENE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SCENE_REGISTER_GET BT_MESH_MODEL_OP_SCENE_REGISTER_GET
+#define ESP_BLE_MESH_MODEL_OP_SCENE_REGISTER_STATUS BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS
+
+/*!< Scene Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_SCENE_STORE BT_MESH_MODEL_OP_SCENE_STORE
+#define ESP_BLE_MESH_MODEL_OP_SCENE_STORE_UNACK BT_MESH_MODEL_OP_SCENE_STORE_UNACK
+#define ESP_BLE_MESH_MODEL_OP_SCENE_DELETE BT_MESH_MODEL_OP_SCENE_DELETE
+#define ESP_BLE_MESH_MODEL_OP_SCENE_DELETE_UNACK BT_MESH_MODEL_OP_SCENE_DELETE_UNACK
+
+/*!< Scheduler Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_GET BT_MESH_MODEL_OP_SCHEDULER_ACT_GET
+#define ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_STATUS BT_MESH_MODEL_OP_SCHEDULER_ACT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_SCHEDULER_GET BT_MESH_MODEL_OP_SCHEDULER_GET
+#define ESP_BLE_MESH_MODEL_OP_SCHEDULER_STATUS BT_MESH_MODEL_OP_SCHEDULER_STATUS
+
+/*!< Scheduler Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_SET BT_MESH_MODEL_OP_SCHEDULER_ACT_SET
+#define ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_SET_UNACK BT_MESH_MODEL_OP_SCHEDULER_ACT_SET_UNACK
+
+typedef uint32_t esp_ble_mesh_light_message_opcode_t; /*!< esp_ble_mesh_light_message_opcode_t belongs to esp_ble_mesh_opcode_t,
+ this typedef is only used to locate the opcodes used by functions
+ esp_ble_mesh_light_client_get_state & esp_ble_mesh_light_client_set_state */
+/*!< Light Lightness Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS
+
+/*!< Light Lightness Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK
+
+/*!< Light CTL Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_GET BT_MESH_MODEL_OP_LIGHT_CTL_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_SET BT_MESH_MODEL_OP_LIGHT_CTL_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK BT_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS BT_MESH_MODEL_OP_LIGHT_CTL_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS
+
+/*!< Light CTL Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK
+
+/*!< Light HSL Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_GET BT_MESH_MODEL_OP_LIGHT_HSL_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_GET BT_MESH_MODEL_OP_LIGHT_HSL_HUE_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS BT_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SET BT_MESH_MODEL_OP_LIGHT_HSL_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK BT_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS BT_MESH_MODEL_OP_LIGHT_HSL_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS
+
+/*!< Light HSL Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK /* Model spec is wrong */
+
+/*!< Light xyL Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_GET BT_MESH_MODEL_OP_LIGHT_XYL_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_SET BT_MESH_MODEL_OP_LIGHT_XYL_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK BT_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS BT_MESH_MODEL_OP_LIGHT_XYL_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS
+
+/*!< Light xyL Setup Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK
+
+/*!< Light Control Message Opcode */
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_MODE_GET BT_MESH_MODEL_OP_LIGHT_LC_MODE_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS BT_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_OM_GET BT_MESH_MODEL_OP_LIGHT_LC_OM_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET BT_MESH_MODEL_OP_LIGHT_LC_OM_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS BT_MESH_MODEL_OP_LIGHT_LC_OM_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK
+#define ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS
+
+typedef uint32_t esp_ble_mesh_opcode_t;
+/*!< End of defines of esp_ble_mesh_opcode_t */
+
+#define ESP_BLE_MESH_CFG_STATUS_SUCCESS STATUS_SUCCESS
+#define ESP_BLE_MESH_CFG_STATUS_INVALID_ADDRESS STATUS_INVALID_ADDRESS
+#define ESP_BLE_MESH_CFG_STATUS_INVALID_MODEL STATUS_INVALID_MODEL
+#define ESP_BLE_MESH_CFG_STATUS_INVALID_APPKEY STATUS_INVALID_APPKEY
+#define ESP_BLE_MESH_CFG_STATUS_INVALID_NETKEY STATUS_INVALID_NETKEY
+#define ESP_BLE_MESH_CFG_STATUS_INSUFFICIENT_RESOURCES STATUS_INSUFF_RESOURCES
+#define ESP_BLE_MESH_CFG_STATUS_KEY_INDEX_ALREADY_STORED STATUS_IDX_ALREADY_STORED
+#define ESP_BLE_MESH_CFG_STATUS_INVALID_PUBLISH_PARAMETERS STATUS_NVAL_PUB_PARAM
+#define ESP_BLE_MESH_CFG_STATUS_NOT_A_SUBSCRIBE_MODEL STATUS_NOT_SUB_MOD
+#define ESP_BLE_MESH_CFG_STATUS_STORAGE_FAILURE STATUS_STORAGE_FAIL
+#define ESP_BLE_MESH_CFG_STATUS_FEATURE_NOT_SUPPORTED STATUS_FEAT_NOT_SUPP
+#define ESP_BLE_MESH_CFG_STATUS_CANNOT_UPDATE STATUS_CANNOT_UPDATE
+#define ESP_BLE_MESH_CFG_STATUS_CANNOT_REMOVE STATUS_CANNOT_REMOVE
+#define ESP_BLE_MESH_CFG_STATUS_CANNOT_BIND STATUS_CANNOT_BIND
+#define ESP_BLE_MESH_CFG_STATUS_TEMP_UNABLE_TO_CHANGE_STATE STATUS_TEMP_STATE_CHG_FAIL
+#define ESP_BLE_MESH_CFG_STATUS_CANNOT_SET STATUS_CANNOT_SET
+#define ESP_BLE_MESH_CFG_STATUS_UNSPECIFIED_ERROR STATUS_UNSPECIFIED
+#define ESP_BLE_MESH_CFG_STATUS_INVALID_BINDING STATUS_INVALID_BINDING
+typedef uint8_t esp_ble_mesh_cfg_status_t; /*!< This typedef is only used to indicate the status code
+ contained in some of the Config Server Model status message */
+
+#define ESP_BLE_MESH_MODEL_STATUS_SUCCESS 0x00
+#define ESP_BLE_MESH_MODEL_STATUS_CANNOT_SET_RANGE_MIN 0x01
+#define ESP_BLE_MESH_MODEL_STATUS_CANNOT_SET_RANGE_MAX 0x02
+typedef uint8_t esp_ble_mesh_model_status_t; /*!< This typedef is only used to indicate the status code contained in
+ some of the server model (e.g. Generic Server Model) status message */
+
+/** @def ESP_BLE_MESH_TRANSMIT
+ *
+ * @brief Encode transmission count & interval steps.
+ *
+ * @note For example, ESP_BLE_MESH_TRANSMIT(2, 20) means that the message
+ * will be sent about 90ms(count is 3, step is 1, interval is 30 ms
+ * which includes 10ms of advertising interval random delay).
+ *
+ * @param count Number of retransmissions (first transmission is excluded).
+ * @param int_ms Interval steps in milliseconds. Must be greater than 0
+ * and a multiple of 10.
+ *
+ * @return BLE Mesh transmit value that can be used e.g. for the default
+ * values of the Configuration Model data.
+ */
+#define ESP_BLE_MESH_TRANSMIT(count, int_ms) BT_MESH_TRANSMIT(count, int_ms)
+
+/** @def ESP_BLE_MESH_GET_TRANSMIT_COUNT
+ *
+ * @brief Decode transmit count from a transmit value.
+ *
+ * @param transmit Encoded transmit count & interval value.
+ *
+ * @return Transmission count (actual transmissions equal to N + 1).
+ */
+#define ESP_BLE_MESH_GET_TRANSMIT_COUNT(transmit) BT_MESH_TRANSMIT_COUNT(transmit)
+
+/** @def ESP_BLE_MESH_GET_TRANSMIT_INTERVAL
+ *
+ * @brief Decode transmit interval from a transmit value.
+ *
+ * @param transmit Encoded transmit count & interval value.
+ *
+ * @return Transmission interval in milliseconds.
+ */
+#define ESP_BLE_MESH_GET_TRANSMIT_INTERVAL(transmit) BT_MESH_TRANSMIT_INT(transmit)
+
+/** @def ESP_BLE_MESH_PUBLISH_TRANSMIT
+ *
+ * @brief Encode Publish Retransmit count & interval steps.
+ *
+ * @param count Number of retransmissions (first transmission is excluded).
+ * @param int_ms Interval steps in milliseconds. Must be greater than 0
+ * and a multiple of 50.
+ *
+ * @return BLE Mesh transmit value that can be used e.g. for the default
+ * values of the Configuration Model data.
+ */
+#define ESP_BLE_MESH_PUBLISH_TRANSMIT(count, int_ms) BT_MESH_PUB_TRANSMIT(count, int_ms)
+
+/** @def ESP_BLE_MESH_GET_PUBLISH_TRANSMIT_COUNT
+ *
+ * @brief Decode Publish Retransmit count from a given value.
+ *
+ * @param transmit Encoded Publish Retransmit count & interval value.
+ *
+ * @return Retransmission count (actual transmissions equal to N + 1).
+ */
+#define ESP_BLE_MESH_GET_PUBLISH_TRANSMIT_COUNT(transmit) BT_MESH_PUB_TRANSMIT_COUNT(transmit)
+
+/** @def ESP_BLE_MESH_GET_PUBLISH_TRANSMIT_INTERVAL
+ *
+ * @brief Decode Publish Retransmit interval from a given value.
+ *
+ * @param transmit Encoded Publish Retransmit count & interval value.
+ *
+ * @return Transmission interval in milliseconds.
+ */
+#define ESP_BLE_MESH_GET_PUBLISH_TRANSMIT_INTERVAL(transmit) BT_MESH_PUB_TRANSMIT_INT(transmit)
+
+/* esp_ble_mesh_cb_t is not needed to be initialized by users (set with 0 and will be initialized internally) */
+typedef uint32_t esp_ble_mesh_cb_t;
+
+typedef enum {
+ ESP_BLE_MESH_TYPE_PROV_CB,
+ ESP_BLE_MESH_TYPE_OUTPUT_NUM_CB,
+ ESP_BLE_MESH_TYPE_OUTPUT_STR_CB,
+ ESP_BLE_MESH_TYPE_INTPUT_CB,
+ ESP_BLE_MESH_TYPE_LINK_OPEN_CB,
+ ESP_BLE_MESH_TYPE_LINK_CLOSE_CB,
+ ESP_BLE_MESH_TYPE_COMPLETE_CB,
+ ESP_BLE_MESH_TYPE_RESET_CB,
+} esp_ble_mesh_cb_type_t;
+
+/*!< This enum value is provisioning authentication oob method */
+typedef enum {
+ ESP_BLE_MESH_NO_OOB,
+ ESP_BLE_MESH_STATIC_OOB,
+ ESP_BLE_MESH_OUTPUT_OOB,
+ ESP_BLE_MESH_INPUT_OOB,
+} esp_ble_mesh_oob_method_t;
+
+/*!< This enum value is associated with bt_mesh_output_action_t in mesh_main.h */
+typedef enum {
+ ESP_BLE_MESH_NO_OUTPUT = 0,
+ ESP_BLE_MESH_BLINK = BIT(0),
+ ESP_BLE_MESH_BEEP = BIT(1),
+ ESP_BLE_MESH_VIBRATE = BIT(2),
+ ESP_BLE_MESH_DISPLAY_NUMBER = BIT(3),
+ ESP_BLE_MESH_DISPLAY_STRING = BIT(4),
+} esp_ble_mesh_output_action_t;
+
+/*!< This enum value is associated with bt_mesh_input_action_t in mesh_main.h */
+typedef enum {
+ ESP_BLE_MESH_NO_INPUT = 0,
+ ESP_BLE_MESH_PUSH = BIT(0),
+ ESP_BLE_MESH_TWIST = BIT(1),
+ ESP_BLE_MESH_ENTER_NUMBER = BIT(2),
+ ESP_BLE_MESH_ENTER_STRING = BIT(3),
+} esp_ble_mesh_input_action_t;
+
+/*!< This enum value is associated with bt_mesh_prov_bearer_t in mesh_main.h */
+typedef enum {
+ ESP_BLE_MESH_PROV_ADV = BIT(0),
+ ESP_BLE_MESH_PROV_GATT = BIT(1),
+} esp_ble_mesh_prov_bearer_t;
+
+#define ESP_BLE_MESH_MODEL_OP_1(b0) BT_MESH_MODEL_OP_1(b0)
+#define ESP_BLE_MESH_MODEL_OP_2(b0, b1) BT_MESH_MODEL_OP_2(b0, b1)
+#define ESP_BLE_MESH_MODEL_OP_3(b0, cid) BT_MESH_MODEL_OP_3(b0, cid)
+
+/*!< This macro is associated with BT_MESH_MODEL in mesh_access.h */
+#define ESP_BLE_MESH_SIG_MODEL(_id, _op, _pub, _user_data) \
+{ \
+ .model_id = (_id), \
+ .op = _op, \
+ .keys = { [0 ... (CONFIG_BT_MESH_MODEL_KEY_COUNT - 1)] = \
+ ESP_BLE_MESH_KEY_UNUSED }, \
+ .pub = _pub, \
+ .groups = { [0 ... (CONFIG_BT_MESH_MODEL_GROUP_COUNT - 1)] = \
+ ESP_BLE_MESH_ADDR_UNASSIGNED }, \
+ .user_data = _user_data, \
+}
+
+/*!< This macro is associated with BT_MESH_MODEL_VND in mesh_access.h */
+#define ESP_BLE_MESH_VENDOR_MODEL(_company, _id, _op, _pub, _user_data) \
+{ \
+ .vnd.company_id = (_company), \
+ .vnd.model_id = (_id), \
+ .op = _op, \
+ .pub = _pub, \
+ .keys = { [0 ... (CONFIG_BT_MESH_MODEL_KEY_COUNT - 1)] = \
+ ESP_BLE_MESH_KEY_UNUSED }, \
+ .groups = { [0 ... (CONFIG_BT_MESH_MODEL_GROUP_COUNT - 1)] = \
+ ESP_BLE_MESH_ADDR_UNASSIGNED }, \
+ .user_data = _user_data, \
+}
+
+/** @brief Helper to define a BLE Mesh element within an array.
+ *
+ * In case the element has no SIG or Vendor models, the helper
+ * macro ESP_BLE_MESH_MODEL_NONE can be given instead.
+ *
+ * @note This macro is associated with BT_MESH_ELEM in mesh_access.h
+ *
+ * @param _loc Location Descriptor.
+ * @param _mods Array of SIG models.
+ * @param _vnd_mods Array of vendor models.
+ */
+#define ESP_BLE_MESH_ELEMENT(_loc, _mods, _vnd_mods) \
+{ \
+ .location = (_loc), \
+ .sig_model_count = ARRAY_SIZE(_mods), \
+ .sig_models = (_mods), \
+ .vnd_model_count = ARRAY_SIZE(_vnd_mods), \
+ .vnd_models = (_vnd_mods), \
+}
+
+#define ESP_BLE_MESH_PROV(uuid, sta_val, sta_val_len, out_size, out_act, in_size, in_act) { \
+ .uuid = uuid, \
+ .static_val = sta_val, \
+ .static_val_len = sta_val_len, \
+ .output_size = out_size, \
+ .output_action = out_act, \
+ .input_size = in_size, \
+ .input_action = in_act, \
+}
+
+typedef struct esp_ble_mesh_model esp_ble_mesh_model_t;
+
+/*!< Abstraction that describes a BLE Mesh Element.
+ This structure is associated with bt_mesh_elem in mesh_access.h */
+typedef struct {
+ /* Element Address, assigned during provisioning. */
+ uint16_t element_addr;
+
+ /* Location Descriptor (GATT Bluetooth Namespace Descriptors) */
+ const uint16_t location;
+
+ /* Model count */
+ const uint8_t sig_model_count;
+ const uint8_t vnd_model_count;
+
+ /* Models */
+ esp_ble_mesh_model_t *sig_models;
+ esp_ble_mesh_model_t *vnd_models;
+} esp_ble_mesh_elem_t;
+
+/*!< Model publication context.
+ This structure is associated with bt_mesh_model_pub in mesh_access.h */
+typedef struct {
+ /** The model to which the context belongs. Initialized by the stack. */
+ esp_ble_mesh_model_t *model;
+
+ uint16_t publish_addr; /**< Publish Address. */
+ uint16_t app_idx; /**< Publish AppKey Index. */
+
+ uint8_t ttl; /**< Publish Time to Live. */
+ uint8_t retransmit; /**< Retransmit Count & Interval Steps. */
+
+ uint8_t period; /*!< Publish Period. */
+ uint8_t period_div: 4, /*!< Divisor for the Period. */
+ cred: 1, /*!< Friendship Credentials Flag. */
+ count: 3; /*!< Retransmissions left. */
+
+ uint32_t period_start; /**< Start of the current period. */
+ /** @brief Publication buffer, containing the publication message.
+ *
+ * This can be correctly created using NET_BUF_SIMPLE when the
+ * publication context is defined.
+ *
+ */
+ struct net_buf_simple *msg;
+
+ /* The callback is only used for the BLE Mesh stack, not for the app layer. */
+ esp_ble_mesh_cb_t update;
+
+ /* Role of the device that is going to publish messages */
+ u8_t dev_role;
+
+ /** Publish Period Timer. Only for stack-internal use. */
+ struct k_delayed_work timer;
+} esp_ble_mesh_model_pub_t;
+
+/*!< Model operation context.
+ This structure is associated with bt_mesh_model_op in mesh_access.h */
+typedef struct {
+ const uint32_t opcode; /* Opcode encoded with the ESP_BLE_MESH_MODEL_OP_* macro */
+ const size_t min_len; /* Minimum required message length */
+ esp_ble_mesh_cb_t param_cb; /* The callback is only used for BLE Mesh stack, not for the app layer. */
+} esp_ble_mesh_model_op_t;
+
+/** Define the terminator for the model operation table, each
+ * model operation struct array must use this terminator as
+ * the end tag of the operation unit.
+ */
+#define ESP_BLE_MESH_MODEL_OP_END {0, 0, 0}
+
+/** Abstraction that describes a Mesh Model instance.
+ * This structure is associated with bt_mesh_model in mesh_access.h
+ */
+struct esp_ble_mesh_model {
+ /* Model ID */
+ union {
+ const uint16_t model_id;
+ struct {
+ uint16_t company_id;
+ uint16_t model_id;
+ } vnd;
+ };
+
+ /* The Element to which this Model belongs */
+ esp_ble_mesh_elem_t *element;
+
+ /* Model Publication */
+ esp_ble_mesh_model_pub_t *const pub;
+
+ /* AppKey List */
+ uint16_t keys[CONFIG_BT_MESH_MODEL_KEY_COUNT];
+
+ /* Subscription List (group or virtual addresses) */
+ uint16_t groups[CONFIG_BT_MESH_MODEL_GROUP_COUNT];
+
+ /* Model operation context */
+ esp_ble_mesh_model_op_t *op;
+
+ /* Model-specific user data */
+ void *user_data;
+
+ /* Internal information, mainly for persistent storage */
+ uint8_t element_idx; /* Belongs to Nth element */
+ uint8_t model_idx; /* Is the Nth model in the element */
+ uint16_t flags; /* Information about what has changed */
+};
+
+/** Helper to define an empty model array.
+ * This structure is associated with BT_MESH_MODEL_NONE in mesh_access.h
+ */
+#define ESP_BLE_MESH_MODEL_NONE ((esp_ble_mesh_model_t []){})
+
+/** Message sending context.
+ * This structure is associated with bt_mesh_msg_ctx in mesh_access.h
+ */
+typedef struct {
+ /** NetKey Index of the subnet through which to send the message. */
+ uint16_t net_idx;
+
+ /** AppKey Index for message encryption. */
+ uint16_t app_idx;
+
+ /** Remote address. */
+ uint16_t addr;
+
+ /** Received TTL value. Not used for sending. */
+ uint8_t recv_ttl: 7;
+
+ /** Force sending reliably by using segment acknowledgement */
+ uint8_t send_rel: 1;
+
+ /** TTL, or BT_MESH_TTL_DEFAULT for default TTL. */
+ uint8_t send_ttl;
+
+ /** Opcode of a received message. Not used for sending message. */
+ uint32_t recv_op;
+
+ /** DST of a received message. Not used for sending message. */
+ uint16_t recv_dst;
+
+ /** Model corresponding to the message, no need to be initialized before sending message */
+ esp_ble_mesh_model_t *model;
+
+ /** Indicate if the message is sent by a node server model, no need to be initialized before sending message */
+ bool srv_send;
+} esp_ble_mesh_msg_ctx_t;
+
+/** Provisioning properties & capabilities.
+ * This structure is associated with bt_mesh_prov in mesh_access.h
+ */
+typedef struct {
+#if CONFIG_BT_MESH_NODE
+ /** The UUID that is used when advertising as an unprovisioned device */
+ const uint8_t *uuid;
+
+ /** Flag indicates whether unprovisioned devices support OOB public key */
+ bool oob_pub_key;
+
+ /* This callback is only used for the BLE Mesh stack, not for the app layer */
+ esp_ble_mesh_cb_t oob_pub_key_cb;
+
+ /** Static OOB value */
+ const uint8_t *static_val;
+ /** Static OOB value length */
+ uint8_t static_val_len;
+
+ /** Maximum size of Output OOB supported */
+ uint8_t output_size;
+ /** Supported Output OOB Actions */
+ uint16_t output_actions;
+
+ /** Maximum size of Input OOB supported */
+ uint8_t input_size;
+ /** Supported Input OOB Actions */
+ uint16_t input_actions;
+
+ /* These callbacks are only used for the BLE Mesh stack, not for the app layer */
+ esp_ble_mesh_cb_t output_num_cb;
+ esp_ble_mesh_cb_t output_str_cb;
+ esp_ble_mesh_cb_t input_cb;
+ esp_ble_mesh_cb_t link_open_cb;
+ esp_ble_mesh_cb_t link_close_cb;
+ esp_ble_mesh_cb_t complete_cb;
+ esp_ble_mesh_cb_t reset_cb;
+#endif /* CONFIG_BT_MESH_NODE */
+#ifdef CONFIG_BT_MESH_PROVISIONER
+ /* Provisioner device UUID */
+ const uint8_t *prov_uuid;
+
+ /* Primary element address of the provisioner */
+ const uint16_t prov_unicast_addr;
+
+ /* Pre-incremental unicast address value to be assigned to the first device */
+ uint16_t prov_start_address;
+
+ /* Attention timer contained in Provisioning Invite PDU */
+ uint8_t prov_attention;
+
+ /* Provisioning Algorithm for the Provisioner */
+ uint8_t prov_algorithm;
+
+ /* Provisioner public key oob */
+ uint8_t prov_pub_key_oob;
+
+ /* The callback is only used for BLE Mesh stack, not for the app layer */
+ esp_ble_mesh_cb_t provisioner_prov_read_oob_pub_key;
+
+ /* Provisioner static oob value */
+ uint8_t *prov_static_oob_val;
+ /* Provisioner static oob value length */
+ uint8_t prov_static_oob_len;
+
+ /* These callbacks are only used for BLE Mesh stack, not for the app layer */
+ esp_ble_mesh_cb_t provisioner_prov_input;
+ esp_ble_mesh_cb_t provisioner_prov_output;
+
+ /* Key refresh and IV update flag */
+ uint8_t flags;
+
+ /* IV index */
+ uint32_t iv_index;
+
+ /* These callbacks are only used for BLE Mesh stack, not for the app layer */
+ esp_ble_mesh_cb_t provisioner_link_open;
+ esp_ble_mesh_cb_t provisioner_link_close;
+ esp_ble_mesh_cb_t provisioner_prov_comp;
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+} esp_ble_mesh_prov_t;
+
+/** Node Composition
+ * This structure is associated with bt_mesh_comp in mesh_access.h
+ */
+typedef struct {
+ uint16_t cid;
+ uint16_t pid;
+ uint16_t vid;
+
+ size_t element_count;
+ esp_ble_mesh_elem_t *elements;
+} esp_ble_mesh_comp_t;
+
+typedef enum {
+ ROLE_NODE = 0,
+ ROLE_PROVISIONER,
+ ROLE_FAST_PROV,
+} esp_ble_mesh_dev_role_t;
+
+typedef struct {
+ esp_ble_mesh_opcode_t opcode; /*!< Message opcode */
+ esp_ble_mesh_model_t *model; /*!< Pointer to the client model structure */
+ esp_ble_mesh_msg_ctx_t ctx; /*!< The context used to send message */
+ int32_t msg_timeout; /*!< Timeout value (ms) to get response to the sent message */
+ /*!< Note: if using default timeout value in menuconfig, make sure to set this value to 0 */
+ uint8_t msg_role; /*!< Role of the device - Node/Provisioner, only used for tx */
+} esp_ble_mesh_client_common_param_t;
+
+typedef uint8_t esp_ble_mesh_dev_add_flag_t;
+#define ADD_DEV_RM_AFTER_PROV_FLAG BIT(0)
+#define ADD_DEV_START_PROV_NOW_FLAG BIT(1)
+#define ADD_DEV_FLUSHABLE_DEV_FLAG BIT(2)
+typedef struct {
+ esp_bd_addr_t addr;
+ esp_ble_addr_type_t addr_type;
+ uint8_t uuid[16];
+ uint16_t oob_info;
+ /*!< ADD_DEV_START_PROV_NOW_FLAG shall not be set if the bearer has both PB-ADV and PB-GATT enabled */
+ esp_ble_mesh_prov_bearer_t bearer;
+} esp_ble_mesh_unprov_dev_add_t;
+
+#define DEL_DEV_ADDR_FLAG BIT(0)
+#define DEL_DEV_UUID_FLAG BIT(1)
+typedef struct {
+ union {
+ struct {
+ esp_bd_addr_t addr;
+ esp_ble_addr_type_t addr_type;
+ };
+ uint8_t uuid[16];
+ };
+ uint8_t flag; /*!< BIT0: device address; BIT1: device UUID */
+} esp_ble_mesh_device_delete_t;
+
+#define PROV_DATA_NET_IDX_FLAG BIT(0)
+#define PROV_DATA_FLAGS_FLAG BIT(1)
+#define PROV_DATA_IV_INDEX_FLAG BIT(2)
+typedef struct {
+ union {
+ uint16_t net_idx;
+ uint8_t flags;
+ uint32_t iv_index;
+ };
+ uint8_t flag; /*!< BIT0: net_idx; BIT1: flags; BIT2: iv_index */
+} esp_ble_mesh_prov_data_info_t;
+
+typedef struct {
+ uint16_t unicast_min; /* Minimum unicast address used for fast provisioning */
+ uint16_t unicast_max; /* Maximum unicast address used for fast provisioning */
+ uint16_t net_idx; /* Netkey index used for fast provisioning */
+ uint8_t flags; /* Flags used for fast provisioning */
+ uint32_t iv_index; /* IV Index used for fast provisioning */
+ uint8_t offset; /* Offset of the UUID to be compared */
+ uint8_t match_len; /* Length of the UUID to be compared */
+ uint8_t match_val[16]; /* Value of UUID to be compared */
+} esp_ble_mesh_fast_prov_info_t;
+
+typedef enum {
+ FAST_PROV_ACT_NONE,
+ FAST_PROV_ACT_ENTER,
+ FAST_PROV_ACT_SUSPEND,
+ FAST_PROV_ACT_EXIT,
+ FAST_PROV_ACT_MAX,
+} esp_ble_mesh_fast_prov_action_t;
+
+typedef enum {
+ ESP_BLE_MESH_PROV_REGISTER_COMP_EVT, /*!< Initialize BLE Mesh provisioning capabilities and internal data information completion event */
+ ESP_BLE_MESH_NODE_SET_UNPROV_DEV_NAME_COMP_EVT, /*!< Set the unprovisioned device name completion event */
+ ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT, /*!< Enable node provisioning functionality completion event */
+ ESP_BLE_MESH_NODE_PROV_DISABLE_COMP_EVT, /*!< Disable node provisioning functionality completion event */
+ ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT, /*!< Establish a BLE Mesh link event */
+ ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT, /*!< Close a BLE Mesh link event */
+ ESP_BLE_MESH_NODE_PROV_OOB_PUB_KEY_EVT, /*!< Generate Node input OOB public key event */
+ ESP_BLE_MESH_NODE_PROV_OUTPUT_NUMBER_EVT, /*!< Generate Node Output Number event */
+ ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT, /*!< Generate Node Output String event */
+ ESP_BLE_MESH_NODE_PROV_INPUT_EVT, /*!< Event requiring the user to input a number or string */
+ ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT, /*!< Provisioning done event */
+ ESP_BLE_MESH_NODE_PROV_RESET_EVT, /*!< Provisioning reset event */
+ ESP_BLE_MESH_NODE_PROV_SET_OOB_PUB_KEY_COMP_EVT, /*!< Node set oob public key completion event */
+ ESP_BLE_MESH_NODE_PROV_INPUT_NUMBER_COMP_EVT, /*!< Node input number completion event */
+ ESP_BLE_MESH_NODE_PROV_INPUT_STRING_COMP_EVT, /*!< Node input string completion event */
+ ESP_BLE_MESH_NODE_PROXY_IDENTITY_ENABLE_COMP_EVT, /*!< Enable BLE Mesh Proxy Identity advertising completion event */
+ ESP_BLE_MESH_NODE_PROXY_GATT_ENABLE_COMP_EVT, /*!< Enable BLE Mesh GATT Proxy Service completion event */
+ ESP_BLE_MESH_NODE_PROXY_GATT_DISABLE_COMP_EVT, /*!< Disable BLE Mesh GATT Proxy Service completion event */
+#if (CONFIG_BT_MESH_PROVISIONER)
+ ESP_BLE_MESH_PROVISIONER_PROV_ENABLE_COMP_EVT, /*!< Provisioner enable provisioning functionality completion event */
+ ESP_BLE_MESH_PROVISIONER_PROV_DISABLE_COMP_EVT, /*!< Provisioner disable provisioning functionality completion event */
+ ESP_BLE_MESH_PROVISIONER_RECV_UNPROV_ADV_PKT_EVT, /*!< Provisioner receives unprovisioned device beacon event */
+ ESP_BLE_MESH_PROVISIONER_PROV_READ_OOB_PUB_KEY_EVT, /*!< Provisioner read unprovisioned device OOB public key event */
+ ESP_BLE_MESH_PROVISIONER_PROV_INPUT_EVT, /*!< Provisioner input value for provisioning procedure event */
+ ESP_BLE_MESH_PROVISIONER_PROV_OUTPUT_EVT, /*!< Provisioner output value for provisioning procedure event */
+ ESP_BLE_MESH_PROVISIONER_PROV_LINK_OPEN_EVT, /*!< Provisioner establish a BLE Mesh link event */
+ ESP_BLE_MESH_PROVISIONER_PROV_LINK_CLOSE_EVT, /*!< Provisioner close a BLE Mesh link event */
+ ESP_BLE_MESH_PROVISIONER_PROV_COMPLETE_EVT, /*!< Provisioner provisioning done event */
+ ESP_BLE_MESH_PROVISIONER_ADD_UNPROV_DEV_COMP_EVT, /*!< Provisioner add a device to the list which contains devices that are waiting/going to be provisioned completion event */
+ ESP_BLE_MESH_PROVISIONER_DELETE_DEV_COMP_EVT, /*!< Provisioner delete a device from the list, close provisioning link with the device if it exists and remove the device from network completion event */
+ ESP_BLE_MESH_PROVISIONER_SET_DEV_UUID_MATCH_COMP_EVT, /*!< Provisioner set the value to be compared with part of the unprovisioned device UUID completion event */
+ ESP_BLE_MESH_PROVISIONER_SET_PROV_DATA_INFO_COMP_EVT, /*!< Provisioner set net_idx/flags/iv_index used for provisioning completion event */
+ ESP_BLE_MESH_PROVISIONER_PROV_READ_OOB_PUB_KEY_COMP_EVT, /*!< Provisioner read unprovisioned device OOB public key completion event */
+ ESP_BLE_MESH_PROVISIONER_PROV_INPUT_NUMBER_COMP_EVT, /*!< Provisioner input number completion event */
+ ESP_BLE_MESH_PROVISIONER_PROV_INPUT_STRING_COMP_EVT, /*!< Provisioner input string completion event */
+ ESP_BLE_MESH_PROVISIONER_SET_NODE_NAME_COMP_EVT, /*!< Provisioner set node name completion event */
+ ESP_BLE_MESH_PROVISIONER_ADD_LOCAL_APP_KEY_COMP_EVT, /*!< Provisioner add local app key completion event */
+ ESP_BLE_MESH_PROVISIONER_BIND_APP_KEY_TO_MODEL_COMP_EVT, /*!< Provisioner bind local model with local app key completion event */
+ ESP_BLE_MESH_PROVISIONER_ADD_LOCAL_NET_KEY_COMP_EVT, /*!< Provisioner add local network key completion event */
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+ ESP_BLE_MESH_SET_FAST_PROV_INFO_COMP_EVT, /* !< Set fast provisioning information (e.g. unicast address range, net_idx, etc.) completion event */
+ ESP_BLE_MESH_SET_FAST_PROV_ACTION_COMP_EVT, /* !< Set fast provisioning action completion event */
+ ESP_BLE_MESH_PROV_EVT_MAX,
+} esp_ble_mesh_prov_cb_event_t;
+
+typedef enum {
+ ESP_BLE_MESH_MODEL_OPERATION_EVT, /*!< User-defined models receive messages from peer devices (e.g. get, set, status, etc) event */
+ ESP_BLE_MESH_MODEL_SEND_COMP_EVT, /*!< User-defined models send messages completion event */
+ ESP_BLE_MESH_MODEL_PUBLISH_COMP_EVT, /*!< User-defined models publish messages completion event */
+ ESP_BLE_MESH_CLIENT_MODEL_RECV_PUBLISH_MSG_EVT, /*!< User-defined client models receive publish messages event */
+ ESP_BLE_MESH_CLIENT_MODEL_SEND_TIMEOUT_EVT, /*!< Timeout event for the user-defined client models that failed to receive response from peer server models */
+ ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT, /*!< When a model is configured to publish messages periodically, this event will occur during every publish period */
+ ESP_BLE_MESH_MODEL_EVT_MAX,
+} esp_ble_mesh_model_cb_event_t;
+
+typedef union {
+ /**
+ * @brief ESP_BLE_MESH_PROV_REGISTER_COMP_EVT
+ */
+ struct ble_mesh_prov_register_comp_param {
+ int err_code;
+ } prov_register_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_SET_UNPROV_DEV_NAME_COMP_EVT
+ */
+ struct ble_mesh_set_unprov_dev_name_comp_param {
+ int err_code;
+ } node_set_unprov_dev_name_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT
+ */
+ struct ble_mesh_prov_enable_comp_param {
+ int err_code;
+ } node_prov_enable_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_DISABLE_COMP_EVT
+ */
+ struct ble_mesh_prov_disable_comp_param {
+ int err_code;
+ } node_prov_disable_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT
+ */
+ struct ble_mesh_link_open_evt_param {
+ esp_ble_mesh_prov_bearer_t bearer;
+ } node_prov_link_open;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT
+ */
+ struct ble_mesh_link_close_evt_param {
+ esp_ble_mesh_prov_bearer_t bearer;
+ } node_prov_link_close;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_OUTPUT_NUMBER_EVT
+ */
+ struct ble_mesh_output_num_evt_param {
+ esp_ble_mesh_output_action_t action;
+ uint32_t number;
+ } node_prov_output_num;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT
+ */
+ struct ble_mesh_output_str_evt_param {
+ char string[8];
+ } node_prov_output_str;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_INPUT_EVT
+ */
+ struct ble_mesh_input_evt_param {
+ esp_ble_mesh_input_action_t action;
+ uint8_t size;
+ } node_prov_input;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT
+ */
+ struct ble_mesh_provision_complete_evt_param {
+ uint16_t net_idx;
+ uint16_t addr;
+ uint8_t flags;
+ uint32_t iv_index;
+ } node_prov_complete;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_RESET_EVT
+ */
+ struct ble_mesh_provision_reset_param {
+
+ } node_prov_reset;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_SET_OOB_PUB_KEY_COMP_EVT
+ */
+ struct ble_mesh_set_oob_pub_key_comp_param {
+ int err_code;
+ } node_prov_set_oob_pub_key_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_INPUT_NUM_COMP_EVT
+ */
+ struct ble_mesh_input_number_comp_param {
+ int err_code;
+ } node_prov_input_num_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROV_INPUT_STR_COMP_EVT
+ */
+ struct ble_mesh_input_string_comp_param {
+ int err_code;
+ } node_prov_input_str_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROXY_IDENTITY_ENABLE_COMP_EVT
+ */
+ struct ble_mesh_proxy_identity_enable_comp_param {
+ int err_code;
+ } node_proxy_identity_enable_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROXY_GATT_ENABLE_COMP_EVT
+ */
+ struct ble_mesh_proxy_gatt_enable_comp_param {
+ int err_code;
+ } node_proxy_gatt_enable_comp;
+ /**
+ * @brief ESP_BLE_MESH_NODE_PROXY_GATT_DISABLE_COMP_EVT
+ */
+ struct ble_mesh_proxy_gatt_disable_comp_param {
+ int err_code;
+ } node_proxy_gatt_disable_comp;
+#if (CONFIG_BT_MESH_PROVISIONER)
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_RECV_UNPROV_ADV_PKT_EVT
+ */
+ struct ble_mesh_provisioner_recv_unprov_adv_pkt_param {
+ uint8_t dev_uuid[16];
+ uint8_t addr[6];
+ esp_ble_addr_type_t addr_type;
+ uint16_t oob_info;
+ uint8_t adv_type;
+ esp_ble_mesh_prov_bearer_t bearer;
+ } provisioner_recv_unprov_adv_pkt;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_ENABLE_COMP_EVT
+ */
+ struct ble_mesh_provisioner_prov_enable_comp_param {
+ int err_code;
+ } provisioner_prov_enable_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_DISABLE_COMP_EVT
+ */
+ struct ble_mesh_provisioner_prov_disable_comp_param {
+ int err_code;
+ } provisioner_prov_disable_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_LINK_OPEN_EVT
+ */
+ struct ble_mesh_provisioner_link_open_evt_param {
+ esp_ble_mesh_prov_bearer_t bearer;
+ } provisioner_prov_link_open;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_READ_OOB_PUB_KEY_EVT
+ */
+ struct ble_mesh_provisioner_prov_read_oob_pub_key_evt_param {
+ uint8_t link_idx;
+ } provisioner_prov_read_oob_pub_key;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_INPUT_EVT
+ */
+ struct ble_mesh_provisioner_prov_input_evt_param {
+ esp_ble_mesh_oob_method_t method;
+ esp_ble_mesh_output_action_t action;
+ uint8_t size;
+ uint8_t link_idx;
+ } provisioner_prov_input;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_OUTPUT_EVT
+ */
+ struct ble_mesh_provisioner_prov_output_evt_param {
+ esp_ble_mesh_oob_method_t method;
+ esp_ble_mesh_input_action_t action;
+ uint8_t size;
+ uint8_t link_idx;
+ union {
+ char string[8];
+ uint32_t number;
+ };
+ } provisioner_prov_output;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_LINK_CLOSE_EVT
+ */
+ struct ble_mesh_provisioner_link_close_evt_param {
+ esp_ble_mesh_prov_bearer_t bearer;
+ uint8_t reason;
+ } provisioner_prov_link_close;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_COMPLETE_EVT
+ */
+ struct ble_mesh_provisioner_prov_comp_param {
+ int node_idx;
+ esp_ble_mesh_octet16_t device_uuid;
+ uint16_t unicast_addr;
+ uint8_t element_num;
+ uint16_t netkey_idx;
+ } provisioner_prov_complete;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_ADD_UNPROV_DEV_COMP_EVT
+ */
+ struct ble_mesh_provisioner_add_unprov_dev_comp_param {
+ int err_code;
+ } provisioner_add_unprov_dev_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_DELETE_DEV_COMP_EVT
+ */
+ struct ble_mesh_provisioner_delete_dev_comp_param {
+ int err_code;
+ } provisioner_delete_dev_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_SET_DEV_UUID_MATCH_COMP_EVT
+ */
+ struct ble_mesh_provisioner_set_dev_uuid_match_comp_param {
+ int err_code;
+ } provisioner_set_dev_uuid_match_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_SET_PROV_DATA_INFO_COMP_EVT
+ */
+ struct ble_mesh_provisioner_set_prov_data_info_comp_param {
+ int err_code;
+ } provisioner_set_prov_data_info_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_READ_OOB_PUB_KEY_COMP_EVT
+ */
+ struct ble_mesh_provisioner_prov_read_oob_pub_key_comp_param {
+ int err_code;
+ } provisioner_prov_read_oob_pub_key_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_INPUT_NUMBER_COMP_EVT
+ */
+ struct ble_mesh_provisioner_prov_input_num_comp_param {
+ int err_code;
+ } provisioner_prov_input_num_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_PROV_INPUT_STRING_COMP_EVT
+ */
+ struct ble_mesh_provisioner_prov_input_str_comp_param {
+ int err_code;
+ } provisioner_prov_input_str_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_SET_NODE_NAME_COMP_EVT
+ */
+ struct ble_mesh_provisioner_set_node_name_comp_param {
+ int err_code;
+ int node_index;
+ } provisioner_set_node_name_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_ADD_LOCAL_APP_KEY_COMP_EVT
+ */
+ struct ble_mesh_provisioner_add_local_app_key_comp_param {
+ int err_code;
+ uint16_t app_idx;
+ } provisioner_add_app_key_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_BIND_APP_KEY_TO_MODEL_COMP_EVT
+ */
+ struct ble_mesh_provisioner_bind_local_mod_app_comp_param {
+ int err_code;
+ } provisioner_bind_app_key_to_model_comp;
+ /**
+ * @brief ESP_BLE_MESH_PROVISIONER_ADD_LOCAL_NET_KEY_COMP_EVT
+ */
+ struct ble_mesh_provisioner_add_local_net_key_comp_param {
+ int err_code;
+ uint16_t net_idx;
+ } provisioner_add_net_key_comp;
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+#if (CONFIG_BT_MESH_FAST_PROV)
+ struct ble_mesh_set_fast_prov_info_comp_param {
+ uint8_t status_unicast;
+ uint8_t status_net_idx;
+ uint8_t status_match;
+ } set_fast_prov_info_comp;
+ struct ble_mesh_set_fast_prov_action_comp_param {
+ uint8_t status_action;
+ } set_fast_prov_action_comp;
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+} esp_ble_mesh_prov_cb_param_t;
+
+typedef union {
+ /**
+ * @brief ESP_BLE_MESH_MODEL_OPERATION_EVT
+ */
+ struct ble_mesh_model_operation_evt_param {
+ uint32_t opcode;
+ esp_ble_mesh_model_t *model;
+ esp_ble_mesh_msg_ctx_t *ctx;
+ uint16_t length;
+ uint8_t *msg;
+ } model_operation;
+ /**
+ * @brief ESP_BLE_MESH_MODEL_SEND_COMP_EVT
+ */
+ struct ble_mesh_model_send_comp_param {
+ int err_code;
+ uint32_t opcode;
+ esp_ble_mesh_model_t *model;
+ esp_ble_mesh_msg_ctx_t *ctx;
+ } model_send_comp;
+ /**
+ * @brief ESP_BLE_MESH_MODEL_PUBLISH_COMP_EVT
+ */
+ struct ble_mesh_model_publish_comp_param {
+ int err_code;
+ esp_ble_mesh_model_t *model;
+ } model_publish_comp;
+ /**
+ * @brief ESP_BLE_MESH_CLIENT_MODEL_RECV_PUBLISH_MSG_EVT
+ */
+ struct ble_mesh_mod_recv_publish_msg_param {
+ uint32_t opcode;
+ esp_ble_mesh_model_t *model;
+ esp_ble_mesh_msg_ctx_t *ctx;
+ uint16_t length;
+ uint8_t *msg;
+ } client_recv_publish_msg;
+ /**
+ * @brief ESP_BLE_MESH_CLIENT_MODEL_SEND_TIMEOUT_EVT
+ */
+ struct ble_mesh_client_model_send_timeout_param {
+ uint32_t opcode;
+ esp_ble_mesh_model_t *model;
+ esp_ble_mesh_msg_ctx_t *ctx;
+ } client_send_timeout;
+ /**
+ * @brief ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT
+ */
+ struct ble_mesh_model_publish_update_evt_param {
+ esp_ble_mesh_model_t *model;
+ } model_publish_update;
+} esp_ble_mesh_model_cb_param_t;
+
+typedef struct {
+ uint32_t cli_op; /*!< The client message opcode */
+ uint32_t status_op; /*!< The server status opcode corresponding to the client message opcode */
+} esp_ble_mesh_client_op_pair_t;
+
+/*!< Mesh Client Model Context */
+typedef struct {
+ esp_ble_mesh_model_t *model;
+ int op_pair_size; /*!< Size of the op_pair */
+ const esp_ble_mesh_client_op_pair_t *op_pair; /*!< Table containing get/set message opcode and corresponding status message opcode */
+ uint32_t publish_status; /*!< This variable is reserved for BLE Mesh Stack, does not require initializing on the application layer */
+ void *internal_data; /*!< Pointer to the structure of the client model internal data */
+ uint8_t msg_role; /*!< Role of the device (Node/Provisioner) that is going to send messages */
+} esp_ble_mesh_client_t;
+
+#endif /* #if CONFIG_BT_MESH */
+#endif /* _ESP_BLE_MESH_DEFS_H_ */
diff --git a/components/bt/ble_mesh/api/models/esp_ble_mesh_config_model_api.c b/components/bt/ble_mesh/api/models/esp_ble_mesh_config_model_api.c
new file mode 100644
index 0000000000..7a6b395951
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/esp_ble_mesh_config_model_api.c
@@ -0,0 +1,82 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+
+#include "esp_bt_defs.h"
+#include "esp_bt_main.h"
+
+#include "btc_ble_mesh_config_client.h"
+#include "esp_ble_mesh_config_model_api.h"
+
+esp_err_t esp_ble_mesh_register_config_client_callback(esp_ble_mesh_cfg_client_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_CFG_CLIENT, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_register_config_server_callback(esp_ble_mesh_cfg_server_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_CFG_SERVER, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_config_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_cfg_client_get_state_t *get_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !get_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_cfg_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_CFG_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_CONFIG_CLIENT_GET_STATE;
+ arg.cfg_client_get_state.params = params;
+ arg.cfg_client_get_state.get_state = get_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_cfg_client_args_t), btc_ble_mesh_cfg_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_config_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_cfg_client_set_state_t *set_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !set_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_cfg_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_CFG_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_CONFIG_CLIENT_SET_STATE;
+ arg.cfg_client_set_state.params = params;
+ arg.cfg_client_set_state.set_state = set_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_cfg_client_args_t), btc_ble_mesh_cfg_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
diff --git a/components/bt/ble_mesh/api/models/esp_ble_mesh_generic_model_api.c b/components/bt/ble_mesh/api/models/esp_ble_mesh_generic_model_api.c
new file mode 100644
index 0000000000..ca7b55ebf8
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/esp_ble_mesh_generic_model_api.c
@@ -0,0 +1,75 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+
+#include "esp_bt_defs.h"
+#include "esp_bt_main.h"
+
+#include "btc_ble_mesh_generic_client.h"
+#include "esp_ble_mesh_generic_model_api.h"
+
+esp_err_t esp_ble_mesh_register_generic_client_callback(esp_ble_mesh_generic_client_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_GENERIC_CLIENT, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_generic_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_generic_client_get_state_t *get_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !get_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_generic_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_GENERIC_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_GENERIC_CLIENT_GET_STATE;
+ arg.generic_client_get_state.params = params;
+ arg.generic_client_get_state.get_state = get_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_generic_client_args_t), btc_ble_mesh_generic_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_generic_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_generic_client_set_state_t *set_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !set_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_generic_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_GENERIC_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_GENERIC_CLIENT_SET_STATE;
+ arg.generic_client_set_state.params = params;
+ arg.generic_client_set_state.set_state = set_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_generic_client_args_t), btc_ble_mesh_generic_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
diff --git a/components/bt/ble_mesh/api/models/esp_ble_mesh_health_model_api.c b/components/bt/ble_mesh/api/models/esp_ble_mesh_health_model_api.c
new file mode 100644
index 0000000000..4b59c54660
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/esp_ble_mesh_health_model_api.c
@@ -0,0 +1,98 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+
+#include "esp_bt_defs.h"
+#include "esp_bt_main.h"
+
+#include "btc_ble_mesh_health.h"
+#include "esp_ble_mesh_health_model_api.h"
+
+esp_err_t esp_ble_mesh_register_health_client_callback(esp_ble_mesh_health_client_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_HEALTH_CLIENT, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_register_health_server_callback(esp_ble_mesh_health_server_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_HEALTH_SERVER, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_health_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_get_state_t *get_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !get_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_health_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_HEALTH_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_HEALTH_CLIENT_GET_STATE;
+ arg.health_client_get_state.params = params;
+ arg.health_client_get_state.get_state = get_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_health_client_args_t), btc_ble_mesh_health_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_health_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_set_state_t *set_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !set_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_health_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_HEALTH_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_HEALTH_CLIENT_SET_STATE;
+ arg.health_client_set_state.params = params;
+ arg.health_client_set_state.set_state = set_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_health_client_args_t), btc_ble_mesh_health_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_health_server_fault_update(esp_ble_mesh_elem_t *element)
+{
+ btc_ble_mesh_health_server_args_t arg = {0};
+ btc_msg_t msg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_HEALTH_SERVER;
+ msg.act = BTC_BLE_MESH_ACT_HEALTH_SERVER_FAULT_UPDATE;
+ arg.fault_update.element = element;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_health_server_args_t), NULL)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
diff --git a/components/bt/ble_mesh/api/models/esp_ble_mesh_lighting_model_api.c b/components/bt/ble_mesh/api/models/esp_ble_mesh_lighting_model_api.c
new file mode 100644
index 0000000000..82591cb96f
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/esp_ble_mesh_lighting_model_api.c
@@ -0,0 +1,76 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+
+#include "esp_bt_defs.h"
+#include "esp_bt_main.h"
+
+#include "btc_ble_mesh_light_client.h"
+#include "esp_ble_mesh_lighting_model_api.h"
+
+esp_err_t esp_ble_mesh_register_light_client_callback(esp_ble_mesh_light_client_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_LIGHT_CLIENT, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_light_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_light_client_get_state_t *get_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !get_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_light_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_LIGHT_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_LIGHT_CLIENT_GET_STATE;
+ arg.light_client_get_state.params = params;
+ arg.light_client_get_state.get_state = get_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_light_client_args_t), btc_ble_mesh_light_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_light_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_light_client_set_state_t *set_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !set_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_light_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_LIGHT_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_LIGHT_CLIENT_SET_STATE;
+ arg.light_client_set_state.params = params;
+ arg.light_client_set_state.set_state = set_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_light_client_args_t), btc_ble_mesh_light_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
diff --git a/components/bt/ble_mesh/api/models/esp_ble_mesh_sensor_model_api.c b/components/bt/ble_mesh/api/models/esp_ble_mesh_sensor_model_api.c
new file mode 100644
index 0000000000..f967adca4b
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/esp_ble_mesh_sensor_model_api.c
@@ -0,0 +1,76 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+
+#include "esp_bt_defs.h"
+#include "esp_bt_main.h"
+
+#include "btc_ble_mesh_sensor_client.h"
+#include "esp_ble_mesh_sensor_model_api.h"
+
+esp_err_t esp_ble_mesh_register_sensor_client_callback(esp_ble_mesh_sensor_client_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_SENSOR_CLIENT, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_sensor_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_sensor_client_get_state_t *get_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !get_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_sensor_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_SENSOR_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_SENSOR_CLIENT_GET_STATE;
+ arg.sensor_client_get_state.params = params;
+ arg.sensor_client_get_state.get_state = get_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_sensor_client_args_t), btc_ble_mesh_sensor_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_sensor_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_sensor_client_set_state_t *set_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !set_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_sensor_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_SENSOR_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_SENSOR_CLIENT_SET_STATE;
+ arg.sensor_client_set_state.params = params;
+ arg.sensor_client_set_state.set_state = set_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_sensor_client_args_t), btc_ble_mesh_sensor_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
diff --git a/components/bt/ble_mesh/api/models/esp_ble_mesh_time_scene_model_api.c b/components/bt/ble_mesh/api/models/esp_ble_mesh_time_scene_model_api.c
new file mode 100644
index 0000000000..264fa0e1d9
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/esp_ble_mesh_time_scene_model_api.c
@@ -0,0 +1,76 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+
+#include "btc/btc_task.h"
+#include "btc/btc_manage.h"
+
+#include "esp_bt_defs.h"
+#include "esp_bt_main.h"
+
+#include "btc_ble_mesh_time_scene_client.h"
+#include "esp_ble_mesh_time_scene_model_api.h"
+
+esp_err_t esp_ble_mesh_register_time_scene_client_callback(esp_ble_mesh_time_scene_client_cb_t callback)
+{
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ return (btc_profile_cb_set(BTC_PID_TIME_SCENE_CLIENT, callback) == 0 ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_time_scene_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_time_scene_client_get_state_t *get_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !get_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_time_scene_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_TIME_SCENE_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_GET_STATE;
+ arg.time_scene_client_get_state.params = params;
+ arg.time_scene_client_get_state.get_state = get_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_time_scene_client_args_t), btc_ble_mesh_time_scene_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
+esp_err_t esp_ble_mesh_time_scene_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_time_scene_client_set_state_t *set_state)
+{
+ if (!params || !params->model || !params->ctx.addr || !set_state) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ btc_msg_t msg;
+ btc_ble_mesh_time_scene_client_args_t arg;
+
+ ESP_BLUEDROID_STATUS_CHECK(ESP_BLUEDROID_STATUS_ENABLED);
+
+ msg.sig = BTC_SIG_API_CALL;
+ msg.pid = BTC_PID_TIME_SCENE_CLIENT;
+ msg.act = BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_SET_STATE;
+ arg.time_scene_client_set_state.params = params;
+ arg.time_scene_client_set_state.set_state = set_state;
+
+ return (btc_transfer_context(&msg, &arg, sizeof(btc_ble_mesh_time_scene_client_args_t), btc_ble_mesh_time_scene_client_arg_deep_copy)
+ == BT_STATUS_SUCCESS ? ESP_OK : ESP_FAIL);
+}
+
diff --git a/components/bt/ble_mesh/api/models/include/esp_ble_mesh_config_model_api.h b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_config_model_api.h
new file mode 100644
index 0000000000..755e0671b2
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_config_model_api.h
@@ -0,0 +1,670 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_CONFIG_MODEL_API_H_
+#define _ESP_BLE_MESH_CONFIG_MODEL_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+/** @def ESP_BLE_MESH_MODEL_CFG_SRV
+ *
+ * @brief Define a new Config Server Model.
+ *
+ * @note The Config Server Model can only be included by a Primary Element.
+ *
+ * @param srv_data Pointer to a unique Config Server Model user_data.
+ *
+ * @return New Config Server Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_CFG_SRV(srv_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_CONFIG_SRV, \
+ NULL, NULL, srv_data)
+
+/** @def ESP_BLE_MESH_MODEL_CFG_CLI
+ *
+ * @brief Define a new Config Client Model.
+ *
+ * @note The Config Client Model can only be included by a Primary Element.
+ *
+ * @param cli_data Pointer to a unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Config Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_CFG_CLI(cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_CONFIG_CLI, \
+ NULL, NULL, cli_data)
+
+typedef struct esp_ble_mesh_cfg_srv {
+ esp_ble_mesh_model_t *model;
+
+ uint8_t net_transmit; /*!< Network Transmit state */
+ uint8_t relay; /*!< Relay Mode state */
+ uint8_t relay_retransmit; /*!< Relay Retransmit state */
+ uint8_t beacon; /*!< Secure Network Beacon state */
+ uint8_t gatt_proxy; /*!< GATT Proxy state */
+ uint8_t friend_state; /*!< Friend state */
+ uint8_t default_ttl; /*!< Default TTL */
+
+ /** Heartbeat Publication */
+ struct {
+ struct k_delayed_work timer;
+
+ uint16_t dst;
+ uint16_t count;
+ uint8_t period;
+ uint8_t ttl;
+ uint16_t feature;
+ uint16_t net_idx;
+ } heartbeat_pub;
+
+ /** Heartbeat Subscription */
+ struct {
+ int64_t expiry;
+
+ uint16_t src;
+ uint16_t dst;
+ uint16_t count;
+ uint8_t min_hops;
+ uint8_t max_hops;
+
+ /** Optional subscription tracking function */
+ void (*func)(uint8_t hops, uint16_t feature);
+ } heartbeat_sub;
+} esp_ble_mesh_cfg_srv_t;
+
+/** Parameters of Composition Data Get. */
+typedef struct {
+ uint8_t page; /*!< Page number of the Composition Data. */
+} esp_ble_mesh_cfg_composition_data_get_t;
+
+/** Parameters of Model Publication Get. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_pub_get_t;
+
+/** Parameters of SIG Model Subscription Get. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_id; /*!< The model id */
+} esp_ble_mesh_cfg_sig_model_sub_get_t;
+
+/** Parameters of Vendor Model Subscription Get. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_vnd_model_sub_get_t;
+
+/** Parameters of Application Key Get. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+} esp_ble_mesh_cfg_app_key_get_t;
+
+/** Parameters of Node Identity Get. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+} esp_ble_mesh_cfg_node_identity_get_t;
+
+/** Parameters of SIG Model App Get. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_id; /*!< The model id */
+} esp_ble_mesh_cfg_sig_model_app_get_t;
+
+/** Parameters of Vendor Model App Get. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_vnd_model_app_get_t;
+
+/** Parameters of Key Refresh Phase Get. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+} esp_ble_mesh_cfg_kr_phase_get_t;
+
+/** Parameters of Low Power Node PollTimeout Get. */
+typedef struct {
+ uint16_t lpn_addr; /*!< The unicast address of the Low Power node */
+} esp_ble_mesh_cfg_lpn_polltimeout_get_t;
+
+/** Parameters of Beacon Set. */
+typedef struct {
+ uint8_t beacon;
+} esp_ble_mesh_cfg_beacon_set_t;
+
+/** Parameters of Default TTL Set. */
+typedef struct {
+ uint8_t ttl; /*!< The default TTL state value */
+} esp_ble_mesh_cfg_default_ttl_set_t;
+
+/** Parameters of Friend Set. */
+typedef struct {
+ uint8_t friend_state; /*!< The friend state value */
+} esp_ble_mesh_cfg_friend_set_t;
+
+/** Parameters of GATT Proxy Set. */
+typedef struct {
+ uint8_t gatt_proxy; /*!< The GATT Proxy state value */
+} esp_ble_mesh_cfg_gatt_proxy_set_t;
+
+/** Parameters of Relay Set. */
+typedef struct {
+ uint8_t relay; /*!< The relay value */
+ uint8_t relay_retransmit; /*!< The relay retransmit value */
+} esp_ble_mesh_cfg_relay_set_t;
+
+/** Parameters of Network Key Add. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+ uint8_t net_key[16]; /*!< The network key value */
+} esp_ble_mesh_cfg_net_key_add_t;
+
+/** Parameters of Application Key Add. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+ uint16_t app_idx; /*!< The app key index */
+ uint8_t app_key[16]; /*!< The app key value */
+} esp_ble_mesh_cfg_app_key_add_t;
+
+/** Parameters of Model Application Key Bind. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_app_idx; /*!< Index of the app key to bind with the model */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_app_bind_t;
+
+/** Parameters of Model Publication Set. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t publish_addr; /*!< Value of the publish address */
+ uint16_t publish_app_idx; /*!< Index of the application key */
+ bool cred_flag; /*!< Value of the Friendship Credential Flag */
+ uint8_t publish_ttl; /*!< Default TTL value for the publishing messages */
+ uint8_t publish_period; /*!< Period for periodic status publishing */
+ uint8_t publish_retransmit; /*!< Number of retransmissions and number of 50-millisecond steps between retransmissions */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_pub_set_t;
+
+/** Parameters of Model Subscription Add. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t sub_addr; /*!< The address to be added to the Subscription List */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_sub_add_t;
+
+/** Parameters of Model Subscription Delete. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t sub_addr; /*!< The address to be removed from the Subscription List */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_sub_delete_t;
+
+/** Parameters of Model Subscription Overwrite. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t sub_addr; /*!< The address to be added to the Subscription List */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_sub_overwrite_t;
+
+/** Parameters of Model Subscription Virtual Address Add. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint8_t label_uuid[16]; /*!< The Label UUID of the virtual address to be added to the Subscription List */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_sub_va_add_t;
+
+/** Parameters of Model Subscription Virtual Address Delete. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint8_t label_uuid[16]; /*!< The Label UUID of the virtual address to be removed from the Subscription List */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_sub_va_delete_t;
+
+/** Parameters of Model Subscription Virtual Address Overwrite. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint8_t label_uuid[16]; /*!< The Label UUID of the virtual address to be added to the Subscription List */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_sub_va_overwrite_t;
+
+/** Parameters of Model Publication Virtual Address Set. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint8_t label_uuid[16]; /*!< Value of the Label UUID publish address */
+ uint16_t publish_app_idx; /*!< Index of the application key */
+ bool cred_flag; /*!< Value of the Friendship Credential Flag */
+ uint8_t publish_ttl; /*!< Default TTL value for the publishing messages */
+ uint8_t publish_period; /*!< Period for periodic status publishing */
+ uint8_t publish_retransmit; /*!< Number of retransmissions and number of 50-millisecond steps between retransmissions */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_pub_va_set_t;
+
+/** Parameters of Model Subscription Delete All. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_sub_delete_all_t;
+
+/** Parameters of Network Key Update. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+ uint8_t net_key[16]; /*!< The network key value */
+} esp_ble_mesh_cfg_net_key_update_t;
+
+/** Parameters of Network Key Delete. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+} esp_ble_mesh_cfg_net_key_delete_t;
+
+/** Parameters of Application Key Update. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+ uint16_t app_idx; /*!< The app key index */
+ uint8_t app_key[16]; /*!< The app key value */
+} esp_ble_mesh_cfg_app_key_update_t;
+
+/** Parameters of Application Key Delete. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+ uint16_t app_idx; /*!< The app key index */
+} esp_ble_mesh_cfg_app_key_delete_t;
+
+/** Parameters of Node Identity Set. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+ uint8_t identity; /*!< New Node Identity state */
+} esp_ble_mesh_cfg_node_identity_set_t;
+
+/** Parameters of Model Application Key Unbind. */
+typedef struct {
+ uint16_t element_addr; /*!< The element address */
+ uint16_t model_app_idx; /*!< Index of the app key to bind with the model */
+ uint16_t model_id; /*!< The model id */
+ uint16_t company_id; /*!< The company id, if not a vendor model, shall set to 0xFFFF */
+} esp_ble_mesh_cfg_model_app_unbind_t;
+
+/** Parameters of Key Refresh Phase Set. */
+typedef struct {
+ uint16_t net_idx; /*!< The network key index */
+ uint8_t transition; /*!< New Key Refresh Phase Transition */
+} esp_ble_mesh_cfg_kr_phase_set_t;
+
+/** Parameters of Network Transmit Set. */
+typedef struct {
+ uint8_t net_transmit; /*!< Network Transmit State */
+} esp_ble_mesh_cfg_net_transmit_set_t;
+
+/** Parameters of Model Heartbeat Publication Set. */
+typedef struct {
+ uint16_t dst;
+ uint8_t count;
+ uint8_t period;
+ uint8_t ttl;
+ uint16_t feature;
+ uint16_t net_idx;
+} esp_ble_mesh_cfg_heartbeat_pub_set_t;
+
+/** Parameters of Model Heartbeat Subscription Set. */
+typedef struct {
+ uint16_t src;
+ uint16_t dst;
+ uint8_t period;
+} esp_ble_mesh_cfg_heartbeat_sub_set_t;
+
+/**
+ * @brief For ESP_BLE_MESH_MODEL_OP_BEACON_GET
+ * ESP_BLE_MESH_MODEL_OP_COMPOSITION_DATA_GET
+ * ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_GET
+ * ESP_BLE_MESH_MODEL_OP_GATT_PROXY_GET
+ * ESP_BLE_MESH_MODEL_OP_RELAY_GET
+ * ESP_BLE_MESH_MODEL_OP_MODEL_PUB_GET
+ * ESP_BLE_MESH_MODEL_OP_FRIEND_GET
+ * ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_GET
+ * ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_GET
+ * the get_state parameter in the esp_ble_mesh_config_client_get_state function should not be set to NULL.
+ */
+typedef union {
+ esp_ble_mesh_cfg_model_pub_get_t model_pub_get; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_PUB_GET. */
+ esp_ble_mesh_cfg_composition_data_get_t comp_data_get; /*!< For ESP_BLE_MESH_MODEL_OP_COMPOSITION_DATA_GET. */
+ esp_ble_mesh_cfg_sig_model_sub_get_t sig_model_sub_get; /*!< For ESP_BLE_MESH_MODEL_OP_SIG_MODEL_SUB_GET */
+ esp_ble_mesh_cfg_vnd_model_sub_get_t vnd_model_sub_get; /*!< For ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_SUB_GET */
+ esp_ble_mesh_cfg_app_key_get_t app_key_get; /*!< For ESP_BLE_MESH_MODEL_OP_APP_KEY_GET. */
+ esp_ble_mesh_cfg_node_identity_get_t node_identity_get; /*!< For ESP_BLE_MESH_MODEL_OP_NODE_IDENTITY_GET. */
+ esp_ble_mesh_cfg_sig_model_app_get_t sig_model_app_get; /*!< For ESP_BLE_MESH_MODEL_OP_SIG_MODEL_APP_GET */
+ esp_ble_mesh_cfg_vnd_model_app_get_t vnd_model_app_get; /*!< For ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_APP_GET */
+ esp_ble_mesh_cfg_kr_phase_get_t kr_phase_get; /*!< For ESP_BLE_MESH_MODEL_OP_KEY_REFRESH_PHASE_GET */
+ esp_ble_mesh_cfg_lpn_polltimeout_get_t lpn_pollto_get; /*!< For ESP_BLE_MESH_MODEL_OP_LPN_POLLTIMEOUT_GET */
+} esp_ble_mesh_cfg_client_get_state_t;
+
+/**
+ * @brief For ESP_BLE_MESH_MODEL_OP_BEACON_SET
+ * ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_SET
+ * ESP_BLE_MESH_MODEL_OP_GATT_PROXY_SET
+ * ESP_BLE_MESH_MODEL_OP_RELAY_SET
+ * ESP_BLE_MESH_MODEL_OP_MODEL_PUB_SET
+ * ESP_BLE_MESH_MODEL_OP_MODEL_SUB_ADD
+ * ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_ADD
+ * ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE
+ * ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_DELETE
+ * ESP_BLE_MESH_MODEL_OP_MODEL_SUB_OVERWRITE
+ * ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_OVERWRITE
+ * ESP_BLE_MESH_MODEL_OP_NET_KEY_ADD
+ * ESP_BLE_MESH_MODEL_OP_APP_KEY_ADD
+ * ESP_BLE_MESH_MODEL_OP_MODEL_APP_BIND
+ * ESP_BLE_MESH_MODEL_OP_NODE_RESET
+ * ESP_BLE_MESH_MODEL_OP_FRIEND_SET
+ * ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_SET
+ * ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_SET
+ * the set_state parameter in the esp_ble_mesh_config_client_set_state function should not be set to NULL.
+ */
+typedef union {
+ esp_ble_mesh_cfg_beacon_set_t beacon_set; /*!< For ESP_BLE_MESH_MODEL_OP_BEACON_SET */
+ esp_ble_mesh_cfg_default_ttl_set_t default_ttl_set; /*!< For ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_SET */
+ esp_ble_mesh_cfg_friend_set_t friend_set; /*!< For ESP_BLE_MESH_MODEL_OP_FRIEND_SET */
+ esp_ble_mesh_cfg_gatt_proxy_set_t gatt_proxy_set; /*!< For ESP_BLE_MESH_MODEL_OP_GATT_PROXY_SET */
+ esp_ble_mesh_cfg_relay_set_t relay_set; /*!< For ESP_BLE_MESH_MODEL_OP_RELAY_SET */
+ esp_ble_mesh_cfg_net_key_add_t net_key_add; /*!< For ESP_BLE_MESH_MODEL_OP_NET_KEY_ADD */
+ esp_ble_mesh_cfg_app_key_add_t app_key_add; /*!< For ESP_BLE_MESH_MODEL_OP_APP_KEY_ADD */
+ esp_ble_mesh_cfg_model_app_bind_t model_app_bind; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_APP_BIND */
+ esp_ble_mesh_cfg_model_pub_set_t model_pub_set; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_PUB_SET */
+ esp_ble_mesh_cfg_model_sub_add_t model_sub_add; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_SUB_ADD */
+ esp_ble_mesh_cfg_model_sub_delete_t model_sub_delete; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE */
+ esp_ble_mesh_cfg_model_sub_overwrite_t model_sub_overwrite; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_SUB_OVERWRITE */
+ esp_ble_mesh_cfg_model_sub_va_add_t model_sub_va_add; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_ADD */
+ esp_ble_mesh_cfg_model_sub_va_delete_t model_sub_va_delete; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_DELETE */
+ esp_ble_mesh_cfg_model_sub_va_overwrite_t model_sub_va_overwrite; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_OVERWRITE */
+ esp_ble_mesh_cfg_heartbeat_pub_set_t heartbeat_pub_set; /*!< For ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_SET */
+ esp_ble_mesh_cfg_heartbeat_sub_set_t heartbeat_sub_set; /*!< For ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_SET */
+ esp_ble_mesh_cfg_model_pub_va_set_t model_pub_va_set; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_PUB_VIRTUAL_ADDR_SET */
+ esp_ble_mesh_cfg_model_sub_delete_all_t model_sub_delete_all; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE_ALL */
+ esp_ble_mesh_cfg_net_key_update_t net_key_update; /*!< For ESP_BLE_MESH_MODEL_OP_NET_KEY_UPDATE */
+ esp_ble_mesh_cfg_net_key_delete_t net_key_delete; /*!< For ESP_BLE_MESH_MODEL_OP_NET_KEY_DELETE */
+ esp_ble_mesh_cfg_app_key_update_t app_key_update; /*!< For ESP_BLE_MESH_MODEL_OP_APP_KEY_UPDATE */
+ esp_ble_mesh_cfg_app_key_delete_t app_key_delete; /*!< For ESP_BLE_MESH_MODEL_OP_APP_KEY_DELETE */
+ esp_ble_mesh_cfg_node_identity_set_t node_identity_set; /*!< For ESP_BLE_MESH_MODEL_OP_NODE_IDENTITY_SET */
+ esp_ble_mesh_cfg_model_app_unbind_t model_app_unbind; /*!< For ESP_BLE_MESH_MODEL_OP_MODEL_APP_UNBIND */
+ esp_ble_mesh_cfg_kr_phase_set_t kr_phase_set; /*!< For ESP_BLE_MESH_MODEL_OP_KEY_REFRESH_PHASE_SET */
+ esp_ble_mesh_cfg_net_transmit_set_t net_transmit_set; /*!< For ESP_BLE_MESH_MODEL_OP_NETWORK_TRANSMIT_SET */
+} esp_ble_mesh_cfg_client_set_state_t;
+
+typedef struct {
+ uint8_t beacon; /*!< Secure Network Beacon state value */
+} esp_ble_mesh_cfg_beacon_status_cb_t;
+
+typedef struct {
+ uint8_t page; /*!< Page number of the Composition Data */
+ struct net_buf_simple *composition_data; /*!< Pointer to Composition Data for the identified page */
+} esp_ble_mesh_cfg_comp_data_status_cb_t;
+
+typedef struct {
+ uint8_t default_ttl; /*!< Default TTL state value */
+} esp_ble_mesh_cfg_default_ttl_status_cb_t;
+
+typedef struct {
+ uint8_t gatt_proxy; /*!< GATT Proxy state value */
+} esp_ble_mesh_cfg_gatt_proxy_status_cb_t;
+
+typedef struct {
+ uint8_t relay; /*!< Relay state value */
+ uint8_t retransmit; /*!< Relay retransmit value(number of retransmissions and number of 10-millisecond steps between retransmissions) */
+} esp_ble_mesh_cfg_relay_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t element_addr; /*!< Address of the element */
+ uint16_t publish_addr; /*!< Value of the publish address */
+ uint16_t app_idx; /*!< Index of the application key */
+ bool cred_flag; /*!< Value of the Friendship Credential Flag */
+ uint8_t ttl; /*!< Default TTL value for the outgoing messages */
+ uint8_t period; /*!< Period for periodic status publishing */
+ uint8_t transmit; /*!< Number of retransmissions and number of 50-millisecond steps between retransmissions */
+ uint16_t company_id; /*!< Company ID */
+ uint16_t model_id; /*!< Model ID */
+} esp_ble_mesh_cfg_model_pub_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t element_addr; /*!< Address of the element */
+ uint16_t sub_addr; /*!< Value of the address */
+ uint16_t company_id; /*!< Company ID */
+ uint16_t model_id; /*!< Model ID */
+} esp_ble_mesh_cfg_model_sub_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t net_idx; /*!< Index of the NetKey */
+} esp_ble_mesh_cfg_net_key_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t net_idx; /*!< Index of the NetKey */
+ uint16_t app_idx; /*!< Index of the application key */
+} esp_ble_mesh_cfg_app_key_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t element_addr; /*!< Address of the element */
+ uint16_t app_idx; /*!< Index of the application key */
+ uint16_t company_id; /*!< Company ID */
+ uint16_t model_id; /*!< Model ID */
+} esp_ble_mesh_cfg_mod_app_status_cb_t;
+
+typedef struct {
+ uint8_t friend_state; /*!< Friend state value */
+} esp_ble_mesh_cfg_friend_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t dst; /*!< Destination address for Heartbeat messages */
+ uint8_t count; /*!< Number of Heartbeat messages remaining to be sent */
+ uint8_t period; /*!< Period for sending Heartbeat messages */
+ uint8_t ttl; /*!< TTL to be used when sending Heartbeat messages */
+ uint16_t features; /*!< Features that trigger Heartbeat messages when changed */
+ uint16_t net_idx; /*!< Index of the NetKey */
+} esp_ble_mesh_cfg_hb_pub_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t src; /*!< Source address for Heartbeat messages */
+ uint16_t dst; /*!< Destination address for Heartbeat messages */
+ uint8_t period; /*!< Remaining Period for processing Heartbeat messages */
+ uint8_t count; /*!< Number of Heartbeat messages received */
+ uint8_t min_hops; /*!< Minimum hops when receiving Heartbeat messages */
+ uint8_t max_hops; /*!< Maximum hops when receiving Heartbeat messages */
+} esp_ble_mesh_cfg_hb_sub_status_cb_t;
+
+typedef struct {
+ uint8_t net_trans_count:3; /*!< Number of transmissions for each Network PDU originating from the node */
+ uint8_t net_trans_step :5; /*!< Maximum hops when receiving Heartbeat messages */
+} esp_ble_mesh_cfg_net_trans_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t element_addr; /*!< Address of the element */
+ uint16_t company_id; /*!< Company ID */
+ uint16_t model_id; /*!< Model ID */
+ struct net_buf_simple *sub_addr; /*!< A block of all addresses from the Subscription List */
+} esp_ble_mesh_cfg_model_sub_list_cb_t;
+
+typedef struct {
+ struct net_buf_simple *net_idx; /*!< A list of NetKey Indexes known to the node */
+} esp_ble_mesh_cfg_net_key_list_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t net_idx; /*!< NetKey Index of the NetKey that the AppKeys are bound to */
+ struct net_buf_simple *app_idx; /*!< A list of AppKey indexes that are bound to the NetKey identified by NetKeyIndex */
+} esp_ble_mesh_cfg_app_key_list_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t net_idx; /*!< Index of the NetKey */
+ uint8_t identity; /*!< Node Identity state */
+} esp_ble_mesh_cfg_node_id_status_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t element_addr; /*!< Address of the element */
+ uint16_t company_id; /*!< Company ID */
+ uint16_t model_id; /*!< Model ID */
+ struct net_buf_simple *app_idx; /*!< All AppKey indexes bound to the Model */
+} esp_ble_mesh_cfg_model_app_list_cb_t;
+
+typedef struct {
+ uint8_t status; /*!< Status Code for the request message */
+ uint16_t net_idx; /*!< Index of the NetKey */
+ uint8_t phase; /*!< Key Refresh Phase state */
+} esp_ble_mesh_cfg_kr_phase_status_cb_t;
+
+typedef struct {
+ uint16_t lpn_addr; /*!< The unicast address of the Low Power node */
+ int32_t poll_timeout; /*!< The current value of the PollTimeout timer of the Low Power node */
+} esp_ble_mesh_cfg_lpn_pollto_status_cb_t;
+
+typedef union {
+ esp_ble_mesh_cfg_beacon_status_cb_t beacon_status; /*!< The beacon status value */
+ esp_ble_mesh_cfg_comp_data_status_cb_t comp_data_status; /*!< The composition data status value */
+ esp_ble_mesh_cfg_default_ttl_status_cb_t default_ttl_status; /*!< The default_ttl status value */
+ esp_ble_mesh_cfg_gatt_proxy_status_cb_t gatt_proxy_status; /*!< The gatt_proxy status value */
+ esp_ble_mesh_cfg_relay_status_cb_t relay_status; /*!< The relay status value */
+ esp_ble_mesh_cfg_model_pub_status_cb_t model_pub_status; /*!< The model publication status value */
+ esp_ble_mesh_cfg_model_sub_status_cb_t model_sub_status; /*!< The model subscription status value */
+ esp_ble_mesh_cfg_net_key_status_cb_t netkey_status; /*!< The netkey status value */
+ esp_ble_mesh_cfg_app_key_status_cb_t appkey_status; /*!< The appkey status value */
+ esp_ble_mesh_cfg_mod_app_status_cb_t model_app_status; /*!< The model app status value */
+ esp_ble_mesh_cfg_friend_status_cb_t friend_status; /*!< The friend status value */
+ esp_ble_mesh_cfg_hb_pub_status_cb_t heartbeat_pub_status; /*!< The heartbeat publication status value */
+ esp_ble_mesh_cfg_hb_sub_status_cb_t heartbeat_sub_status; /*!< The heartbeat subscription status value */
+ esp_ble_mesh_cfg_net_trans_status_cb_t net_transmit_status; /*!< The network transmit status value */
+ esp_ble_mesh_cfg_model_sub_list_cb_t model_sub_list; /*!< The model subscription list value */
+ esp_ble_mesh_cfg_net_key_list_cb_t netkey_list; /*!< The network key index list value */
+ esp_ble_mesh_cfg_app_key_list_cb_t appkey_list; /*!< The application key index list value */
+ esp_ble_mesh_cfg_node_id_status_cb_t node_identity_status; /*!< The node identity status value */
+ esp_ble_mesh_cfg_model_app_list_cb_t model_app_list; /*!< The model application key index list value */
+ esp_ble_mesh_cfg_kr_phase_status_cb_t kr_phase_status; /*!< The key refresh phase status value */
+ esp_ble_mesh_cfg_lpn_pollto_status_cb_t lpn_timeout_status; /*!< The low power node poll timeout status value */
+} esp_ble_mesh_cfg_client_common_cb_param_t;
+
+typedef struct {
+ int error_code; /*!< Appropriate error code */
+ esp_ble_mesh_client_common_param_t *params; /*!< The client common parameters */
+ esp_ble_mesh_cfg_client_common_cb_param_t status_cb; /*!< The config status message callback values */
+} esp_ble_mesh_cfg_client_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_CFG_CLIENT_GET_STATE_EVT,
+ ESP_BLE_MESH_CFG_CLIENT_SET_STATE_EVT,
+ ESP_BLE_MESH_CFG_CLIENT_PUBLISH_EVT,
+ ESP_BLE_MESH_CFG_CLIENT_TIMEOUT_EVT,
+ ESP_BLE_MESH_CFG_CLIENT_EVT_MAX,
+} esp_ble_mesh_cfg_client_cb_event_t;
+
+typedef struct {
+ uint16_t app_idx; /* AppKey Index of the Config AppKey Add */
+} esp_ble_mesh_cfg_srv_app_key_add_cb_t;
+
+typedef union {
+ esp_ble_mesh_cfg_srv_app_key_add_cb_t app_key_add; /* !< The Config AppKey Add event value */
+} esp_ble_mesh_cfg_server_common_cb_param_t;
+
+typedef struct {
+ esp_ble_mesh_model_t *model; /*!< Pointer to the server model structure */
+ esp_ble_mesh_msg_ctx_t ctx; /*!< The context of the received message */
+ esp_ble_mesh_cfg_server_common_cb_param_t status_cb; /*!< The received configuration message callback values */
+} esp_ble_mesh_cfg_server_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_CFG_SERVER_RECV_MSG_EVT,
+ ESP_BLE_MESH_CFG_SERVER_EVT_MAX,
+} esp_ble_mesh_cfg_server_cb_event_t;
+
+/**
+ * @brief Bluetooth Mesh Config Client and Server Model functions.
+ */
+
+/** @brief: event, event code of Config Client Model events; param, parameters of Config Client Model events */
+typedef void (* esp_ble_mesh_cfg_client_cb_t)(esp_ble_mesh_cfg_client_cb_event_t event,
+ esp_ble_mesh_cfg_client_cb_param_t *param);
+
+/** @brief: event, event code of Config Client Model events; param, parameters of Config Client Model events */
+typedef void (* esp_ble_mesh_cfg_server_cb_t)(esp_ble_mesh_cfg_server_cb_event_t event,
+ esp_ble_mesh_cfg_server_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh Config Client Model callback.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_config_client_callback(esp_ble_mesh_cfg_client_cb_t callback);
+
+/**
+ * @brief Register BLE Mesh Config Server Model callback.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_config_server_callback(esp_ble_mesh_cfg_server_cb_t callback);
+
+/**
+ * @brief Get the value of Config Server Model states using the Config Client Model get messages.
+ *
+ * @note If you want to find the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_opcode_config_client_get_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] get_state: Pointer to a union, each kind of opcode corresponds to one structure inside.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_config_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_cfg_client_get_state_t *get_state);
+
+/**
+ * @brief Set the value of the Configuration Server Model states using the Config Client Model set messages.
+ *
+ * @note If you want to find the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_opcode_config_client_set_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] set_state: Pointer to a union, each kind of opcode corresponds to one structure inside.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_config_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_cfg_client_set_state_t *set_state);
+
+#endif /** _ESP_BLE_MESH_CONFIG_MODEL_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/models/include/esp_ble_mesh_generic_model_api.h b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_generic_model_api.h
new file mode 100644
index 0000000000..a8db2852f7
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_generic_model_api.h
@@ -0,0 +1,478 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Generic Client Model APIs.
+ */
+
+#ifndef _ESP_BLE_MESH_GENERIC_MODEL_API_H_
+#define _ESP_BLE_MESH_GENERIC_MODEL_API_H_
+
+#include "generic_client.h"
+#include "esp_ble_mesh_defs.h"
+
+/** @def ESP_BLE_MESH_MODEL_GEN_ONOFF_CLI
+ *
+ * @brief Define a new Generic OnOff Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic OnOff Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic OnOff Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_GEN_ONOFF_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_GEN_LEVEL_CLI
+ *
+ * @brief Define a new Generic Level Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic Level Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic Level Client Model instance.
+ */
+
+#define ESP_BLE_MESH_MODEL_GEN_LEVEL_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_LEVEL_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_GEN_DEF_TRANS_TIME_CLI
+ *
+ * @brief Define a new Generic Default Transition Time Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic Default Transition
+ * Time Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic Default Transition Time Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_GEN_DEF_TRANS_TIME_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_GEN_POWER_ONOFF_CLI
+ *
+ * @brief Define a new Generic Power OnOff Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic Power OnOff Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic Power OnOff Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_GEN_POWER_ONOFF_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_POWER_ONOFF_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_GEN_POWER_LEVEL_CLI
+ *
+ * @brief Define a new Generic Power Level Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic Power Level Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic Power Level Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_GEN_POWER_LEVEL_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_POWER_LEVEL_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_GEN_BATTERY_CLI
+ *
+ * @brief Define a new Generic Battery Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic Battery Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic Battery Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_GEN_BATTERY_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_BATTERY_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_GEN_LOCATION_CLI
+ *
+ * @brief Define a new Generic Location Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic Location Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic Location Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_GEN_LOCATION_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_LOCATION_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_GEN_PROPERTY_CLI
+ *
+ * @brief Define a new Generic Property Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Generic Property Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Generic Location Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_GEN_PROPERTY_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_PROP_CLI, \
+ NULL, cli_pub, cli_data)
+
+/**
+ * @brief Bluetooth Mesh Generic Client Model Get and Set parameters structure.
+ */
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ uint8_t onoff; /* Target value of Generic OnOff state */
+ uint8_t tid; /* Transaction ID */
+ uint8_t trans_time; /* Time to complete state transition (optional) */
+ uint8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_gen_onoff_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ int16_t level; /* Target value of Generic Level state */
+ uint8_t tid; /* Transaction ID */
+ uint8_t trans_time; /* Time to complete state transition (optional) */
+ uint8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_gen_level_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ int32_t level; /* Delta change of Generic Level state */
+ uint8_t tid; /* Transaction ID */
+ uint8_t trans_time; /* Time to complete state transition (optional) */
+ uint8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_gen_delta_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ int16_t delta_level;/* Delta Level step to calculate Move speed for Generic Level state */
+ uint8_t tid; /* Transaction ID */
+ uint8_t trans_time; /* Time to complete state transition (optional) */
+ uint8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_gen_move_set_t;
+
+typedef struct {
+ uint8_t trans_time; /* The value of the Generic Default Transition Time state */
+} esp_ble_mesh_gen_def_trans_time_set_t;
+
+typedef struct {
+ uint8_t onpowerup; /* The value of the Generic OnPowerUp state */
+} esp_ble_mesh_gen_onpowerup_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ uint16_t power; /* Target value of Generic Power Actual state */
+ uint8_t tid; /* Transaction ID */
+ uint8_t trans_time; /* Time to complete state transition (optional) */
+ uint8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_gen_power_level_set_t;
+
+typedef struct {
+ uint16_t power; /* The value of the Generic Power Default state */
+} esp_ble_mesh_gen_power_default_set_t;
+
+typedef struct {
+ uint16_t range_min; /* Value of Range Min field of Generic Power Range state */
+ uint16_t range_max; /* Value of Range Max field of Generic Power Range state */
+} esp_ble_mesh_gen_power_range_set_t;
+
+typedef struct {
+ int32_t global_latitude; /* Global Coordinates (Latitude) */
+ int32_t global_longitude; /* Global Coordinates (Longitude) */
+ int16_t global_altitude; /* Global Altitude */
+} esp_ble_mesh_gen_loc_global_set_t;
+
+typedef struct {
+ int16_t local_north; /* Local Coordinates (North) */
+ int16_t local_east; /* Local Coordinates (East) */
+ int16_t local_altitude; /* Local Altitude */
+ uint8_t floor_number; /* Floor Number */
+ uint16_t uncertainty; /* Uncertainty */
+} esp_ble_mesh_gen_loc_local_set_t;
+
+typedef struct {
+ uint16_t property_id; /* Property ID identifying a Generic User Property */
+} esp_ble_mesh_gen_user_property_get_t;
+
+typedef struct {
+ uint16_t property_id; /* Property ID identifying a Generic User Property */
+ struct net_buf_simple *property_value; /* Raw value for the User Property */
+} esp_ble_mesh_gen_user_property_set_t;
+
+typedef struct {
+ uint16_t property_id; /* Property ID identifying a Generic Admin Property */
+} esp_ble_mesh_gen_admin_property_get_t;
+
+typedef struct {
+ uint16_t property_id; /* Property ID identifying a Generic Admin Property */
+ uint8_t user_access; /* Enumeration indicating user access */
+ struct net_buf_simple *property_value; /* Raw value for the Admin Property */
+} esp_ble_mesh_gen_admin_property_set_t;
+
+typedef struct {
+ uint16_t property_id; /* Property ID identifying a Generic Manufacturer Property */
+} esp_ble_mesh_gen_manufacturer_property_get_t;
+
+typedef struct {
+ uint16_t property_id; /* Property ID identifying a Generic Manufacturer Property */
+ uint8_t user_access; /* Enumeration indicating user access */
+} esp_ble_mesh_gen_manufacturer_property_set_t;
+
+typedef struct {
+ uint16_t property_id; /* A starting Client Property ID present within an element */
+} esp_ble_mesh_gen_client_properties_get_t;
+
+typedef union {
+ esp_ble_mesh_gen_user_property_get_t user_property_get; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_GET */
+ esp_ble_mesh_gen_admin_property_get_t admin_property_get; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET*/
+ esp_ble_mesh_gen_manufacturer_property_get_t manufacturer_property_get; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_SET */
+ esp_ble_mesh_gen_client_properties_get_t client_properties_get; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET */
+} esp_ble_mesh_generic_client_get_state_t;
+
+typedef union {
+ esp_ble_mesh_gen_onoff_set_t onoff_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET & ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK */
+ esp_ble_mesh_gen_level_set_t level_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_LEVEL_SET & ESP_BLE_MESH_MODEL_OP_GEN_LEVEL_SET_UNACK */
+ esp_ble_mesh_gen_delta_set_t delta_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_DELTA_SET & ESP_BLE_MESH_MODEL_OP_GEN_DELTA_SET_UNACK */
+ esp_ble_mesh_gen_move_set_t move_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_MOVE_SET & ESP_BLE_MESH_MODEL_OP_GEN_MOVE_SET_UNACK */
+ esp_ble_mesh_gen_def_trans_time_set_t def_trans_time_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET & ESP_BLE_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET_UNACK */
+ esp_ble_mesh_gen_onpowerup_set_t power_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_ONPOWERUP_SET & ESP_BLE_MESH_MODEL_OP_GEN_ONPOWERUP_SET_UNACK */
+ esp_ble_mesh_gen_power_level_set_t power_level_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_POWER_LEVEL_SET & ESP_BLE_MESH_MODEL_OP_GEN_POWER_LEVEL_SET_UNACK */
+ esp_ble_mesh_gen_power_default_set_t power_default_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET & ESP_BLE_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET_UNACK */
+ esp_ble_mesh_gen_power_range_set_t power_range_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_POWER_RANGE_SET & ESP_BLE_MESH_MODEL_OP_GEN_POWER_RANGE_SET_UNACK */
+ esp_ble_mesh_gen_loc_global_set_t loc_global_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET & ESP_BLE_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET_UNACK */
+ esp_ble_mesh_gen_loc_local_set_t loc_local_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_LOC_LOCAL_SET & ESP_BLE_MESH_MODEL_OP_GEN_LOC_LOCAL_SET_UNACK */
+ esp_ble_mesh_gen_user_property_set_t user_property_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET & ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET_UNACK */
+ esp_ble_mesh_gen_admin_property_set_t admin_property_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET & ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET_UNACK */
+ esp_ble_mesh_gen_manufacturer_property_set_t manufacturer_property_set; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_SET & ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_SET_UNACK */
+} esp_ble_mesh_generic_client_set_state_t;
+
+/**
+ * @brief Bluetooth Mesh Generic Client Model Get and Set callback parameters structure.
+ */
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ uint8_t present_onoff; /* Current value of Generic OnOff state */
+ uint8_t target_onoff; /* Target value of Generic OnOff state (optional) */
+ uint8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_gen_onoff_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ int16_t present_level; /* Current value of Generic Level state */
+ int16_t target_level; /* Target value of the Generic Level state (optional) */
+ uint8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_gen_level_status_cb_t;
+
+typedef struct {
+ uint8_t trans_time; /* The value of the Generic Default Transition Time state */
+} esp_ble_mesh_gen_def_trans_time_status_cb_t;
+
+typedef struct {
+ uint8_t onpowerup; /* The value of the Generic OnPowerUp state */
+} esp_ble_mesh_gen_onpowerup_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ uint16_t present_power; /* Current value of Generic Power Actual state */
+ uint16_t target_power; /* Target value of Generic Power Actual state (optional) */
+ uint8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_gen_power_level_status_cb_t;
+
+typedef struct {
+ uint16_t power; /* The value of the Generic Power Last state */
+} esp_ble_mesh_gen_power_last_status_cb_t;
+
+typedef struct {
+ uint16_t power; /* The value of the Generic Default Last state */
+} esp_ble_mesh_gen_power_default_status_cb_t;
+
+typedef struct {
+ uint8_t status_code; /* Status Code for the request message */
+ uint16_t range_min; /* Value of Range Min field of Generic Power Range state */
+ uint16_t range_max; /* Value of Range Max field of Generic Power Range state */
+} esp_ble_mesh_gen_power_range_status_cb_t;
+
+typedef struct {
+ u32_t battery_level : 8; /* Value of Generic Battery Level state */
+ u32_t time_to_discharge : 24; /* Value of Generic Battery Time to Discharge state */
+ u32_t time_to_charge : 24; /* Value of Generic Battery Time to Charge state */
+ u32_t flags : 8; /* Value of Generic Battery Flags state */
+} esp_ble_mesh_gen_battery_status_cb_t;
+
+typedef struct {
+ int32_t global_latitude; /* Global Coordinates (Latitude) */
+ int32_t global_longitude; /* Global Coordinates (Longitude) */
+ int16_t global_altitude; /* Global Altitude */
+} esp_ble_mesh_gen_loc_global_status_cb_t;
+
+typedef struct {
+ int16_t local_north; /* Local Coordinates (North) */
+ int16_t local_east; /* Local Coordinates (East) */
+ int16_t local_altitude; /* Local Altitude */
+ uint8_t floor_number; /* Floor Number */
+ uint16_t uncertainty; /* Uncertainty */
+} esp_ble_mesh_gen_loc_local_status_cb_t;
+
+typedef struct {
+ struct net_buf_simple *property_ids; /* Buffer contains a sequence of N User Property IDs */
+} esp_ble_mesh_gen_user_properties_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ uint16_t property_id; /* Property ID identifying a Generic User Property */
+ uint8_t user_access; /* Enumeration indicating user access (optional) */
+ struct net_buf_simple *property_value; /* Raw value for the User Property (C.1) */
+} esp_ble_mesh_gen_user_property_status_cb_t;
+
+typedef struct {
+ struct net_buf_simple *property_ids; /* Buffer contains a sequence of N Admin Property IDs */
+} esp_ble_mesh_gen_admin_properties_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ uint16_t property_id; /* Property ID identifying a Generic Admin Property */
+ uint8_t user_access; /* Enumeration indicating user access (optional) */
+ struct net_buf_simple *property_value; /* Raw value for the Admin Property (C.1) */
+} esp_ble_mesh_gen_admin_property_status_cb_t;
+
+typedef struct {
+ struct net_buf_simple *property_ids; /* Buffer contains a sequence of N Manufacturer Property IDs */
+} esp_ble_mesh_gen_manufacturer_properties_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ uint16_t property_id; /* Property ID identifying a Generic Manufacturer Property */
+ uint8_t user_access; /* Enumeration indicating user access (optional) */
+ struct net_buf_simple *property_value; /* Raw value for the Manufacturer Property (C.1) */
+} esp_ble_mesh_gen_manufacturer_property_status_cb_t;
+
+typedef struct {
+ struct net_buf_simple *property_ids; /* Buffer contains a sequence of N Client Property IDs */
+} esp_ble_mesh_gen_client_properties_status_cb_t;
+
+typedef union {
+ esp_ble_mesh_gen_onoff_status_cb_t onoff_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS */
+ esp_ble_mesh_gen_level_status_cb_t level_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_LEVEL_STATUS */
+ esp_ble_mesh_gen_def_trans_time_status_cb_t def_trans_time_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS */
+ esp_ble_mesh_gen_onpowerup_status_cb_t onpowerup_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS */
+ esp_ble_mesh_gen_power_level_status_cb_t power_level_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS */
+ esp_ble_mesh_gen_power_last_status_cb_t power_last_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_POWER_LAST_STATUS */
+ esp_ble_mesh_gen_power_default_status_cb_t power_default_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS */
+ esp_ble_mesh_gen_power_range_status_cb_t power_range_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS */
+ esp_ble_mesh_gen_battery_status_cb_t battery_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_BATTERY_STATUS */
+ esp_ble_mesh_gen_loc_global_status_cb_t location_global_status; /*!< For ESP_BLE_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS */
+ esp_ble_mesh_gen_loc_local_status_cb_t location_local_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS */
+ esp_ble_mesh_gen_user_properties_status_cb_t user_properties_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS */
+ esp_ble_mesh_gen_user_property_status_cb_t user_property_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS */
+ esp_ble_mesh_gen_admin_properties_status_cb_t admin_properties_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS */
+ esp_ble_mesh_gen_admin_property_status_cb_t admin_property_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS */
+ esp_ble_mesh_gen_manufacturer_properties_status_cb_t manufacturer_properties_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTIES_STATUS */
+ esp_ble_mesh_gen_manufacturer_property_status_cb_t manufacturer_property_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_STATUS */
+ esp_ble_mesh_gen_client_properties_status_cb_t client_properties_status; /*!< ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS */
+} esp_ble_mesh_gen_client_status_cb_t;
+
+typedef struct {
+ int error_code; /*!< Appropriate error code */
+ esp_ble_mesh_client_common_param_t *params; /*!< The client common parameters. */
+ esp_ble_mesh_gen_client_status_cb_t status_cb; /*!< The generic status message callback values */
+} esp_ble_mesh_generic_client_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_GENERIC_CLIENT_GET_STATE_EVT,
+ ESP_BLE_MESH_GENERIC_CLIENT_SET_STATE_EVT,
+ ESP_BLE_MESH_GENERIC_CLIENT_PUBLISH_EVT,
+ ESP_BLE_MESH_GENERIC_CLIENT_TIMEOUT_EVT,
+ ESP_BLE_MESH_GENERIC_CLIENT_EVT_MAX,
+} esp_ble_mesh_generic_client_cb_event_t;
+
+/**
+ * @brief Bluetooth Mesh Generic Client Model function.
+ */
+
+/** @brief: event, event code of Generic Client Model events; param, parameters of Generic Client Model events */
+typedef void (* esp_ble_mesh_generic_client_cb_t)(esp_ble_mesh_generic_client_cb_event_t event,
+ esp_ble_mesh_generic_client_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh Generic Client Model callback.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_generic_client_callback(esp_ble_mesh_generic_client_cb_t callback);
+
+/**
+ * @brief Get the value of Generic Server Model states using the Generic Client Model get messages.
+ *
+ * @note If you want to find the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_generic_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] get_state: Pointer to generic get message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_generic_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_generic_client_get_state_t *get_state);
+
+/**
+ * @brief Set the value of Generic Server Model states using the Generic Client Model set messages.
+ *
+ * @note If you want to find the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_generic_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] set_state: Pointer to generic set message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_generic_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_generic_client_set_state_t *set_state);
+
+
+#endif /* _ESP_BLE_MESH_GENERIC_MODEL_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/models/include/esp_ble_mesh_health_model_api.h b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_health_model_api.h
new file mode 100644
index 0000000000..687bdc604e
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_health_model_api.h
@@ -0,0 +1,261 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _ESP_BLE_MESH_HEALTH_MODEL_API_H_
+#define _ESP_BLE_MESH_HEALTH_MODEL_API_H_
+
+#include "esp_ble_mesh_defs.h"
+
+/** @def ESP_BLE_MESH_MODEL_HEALTH_SRV
+ *
+ * @brief Define a new Health Server Model.
+ *
+ * @note The Health Server Model can only be included by a Primary Element.
+ *
+ * @param srv Pointer to the unique struct esp_ble_mesh_health_srv_t.
+ * @param pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ *
+ * @return New Health Server Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_HEALTH_SRV(srv, pub) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_HEALTH_SRV, \
+ NULL, pub, srv)
+
+/** @def ESP_BLE_MESH_MODEL_HEALTH_CLI
+ *
+ * @brief Define a new Health Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Health Client Model.
+ *
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Health Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_HEALTH_CLI(cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_HEALTH_CLI, \
+ NULL, NULL, cli_data)
+
+typedef struct {
+ /* Fetch current faults */
+ int (*fault_get_cur)(esp_ble_mesh_model_t *model, uint8_t *test_id,
+ uint16_t *company_id, uint8_t *faults, uint8_t *fault_count);
+
+ /* Fetch registered faults */
+ int (*fault_get_reg)(esp_ble_mesh_model_t *model, uint16_t company_id,
+ uint8_t *test_id, uint8_t *faults, uint8_t *fault_count);
+
+ /* Clear registered faults */
+ int (*fault_clear)(esp_ble_mesh_model_t *model, uint16_t company_id);
+
+ /* Run a specific test */
+ int (*fault_test)(esp_ble_mesh_model_t *model, uint8_t test_id, uint16_t company_id);
+
+ /* Attention on */
+ void (*attn_on)(esp_ble_mesh_model_t *model);
+
+ /* Attention off */
+ void (*attn_off)(esp_ble_mesh_model_t *model);
+} esp_ble_mesh_health_srv_cb_t;
+
+/** ESP BLE Mesh Health Server Model Context */
+typedef struct {
+ esp_ble_mesh_model_t *model;
+
+ /* Optional callback struct */
+ const esp_ble_mesh_health_srv_cb_t *cb;
+
+ /* Attention Timer state */
+ struct k_delayed_work attn_timer;
+} esp_ble_mesh_health_srv_t;
+
+/** BLE Mesh Health Client Model fault get Context */
+typedef struct {
+ uint16_t company_id; /*!< Bluetooth assigned 16-bit Company ID */
+} esp_ble_mesh_health_fault_get_t;
+
+/** Mesh Health Client Model attention set Context */
+typedef struct {
+ uint8_t attention; /*!< Value of the Attention Timer state */
+} esp_ble_mesh_health_attention_set_t;
+
+/** Mesh Health client Model period set Context */
+typedef struct {
+ uint8_t fast_period_divisor; /*!< Divider for the Publish Period */
+} esp_ble_mesh_health_period_set_t;
+
+/** BLE Mesh Health Client Model fault test Context */
+typedef struct {
+ uint16_t company_id; /*!< Bluetooth assigned 16-bit Company ID */
+ uint8_t test_id; /*!< ID of a specific test to be performed */
+} esp_ble_mesh_health_fault_test_t;
+
+/** BLE Mesh Health Client Model fault clear Context */
+typedef struct {
+ uint16_t company_id; /*!< Bluetooth assigned 16-bit Company ID */
+} esp_ble_mesh_health_fault_clear_t;
+
+/**
+ * @brief For ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_GET
+ * ESP_BLE_MESH_MODEL_OP_ATTENTION_GET
+ * ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_GET
+ * the get_state parameter in the esp_ble_mesh_health_client_get_state function should not be set to NULL.
+ */
+typedef union {
+ esp_ble_mesh_health_fault_get_t fault_get; /*!< For ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_GET. */
+} esp_ble_mesh_health_client_get_state_t;
+
+/**
+ * @brief For ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR
+ * ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR_UNACK
+ * ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST
+ * ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST_UNACK
+ * ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET
+ * ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET_UNACK
+ * ESP_BLE_MESH_MODEL_OP_ATTENTION_SET
+ * ESP_BLE_MESH_MODEL_OP_ATTENTION_SET_UNACK
+ * the set_state parameter in the esp_ble_mesh_health_client_set_state function should not be set to NULL.
+ */
+typedef union {
+ esp_ble_mesh_health_attention_set_t attention_set; /*!< For ESP_BLE_MESH_MODEL_OP_ATTENTION_SET or ESP_BLE_MESH_MODEL_OP_ATTENTION_SET_UNACK. */
+ esp_ble_mesh_health_period_set_t period_set; /*!< For ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET or ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET_UNACK. */
+ esp_ble_mesh_health_fault_test_t fault_test; /*!< For ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST or ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST_UNACK. */
+ esp_ble_mesh_health_fault_clear_t fault_clear; /*!< For ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR or ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR_UNACK. */
+} esp_ble_mesh_health_client_set_state_t;
+
+typedef struct {
+ uint8_t test_id; /*!< ID of a most recently performed test */
+ uint16_t company_id; /*!< Bluetooth assigned 16-bit Company ID */
+ struct net_buf_simple *fault_array; /*!< FaultArray field contains a sequence of 1-octet fault values */
+} esp_ble_mesh_health_current_status_cb_t;
+
+typedef struct {
+ uint8_t test_id; /*!< ID of a most recently performed test */
+ uint16_t company_id; /*!< Bluetooth assigned 16-bit Company ID */
+ struct net_buf_simple *fault_array; /*!< FaultArray field contains a sequence of 1-octet fault values */
+} esp_ble_mesh_health_fault_status_cb_t;
+
+typedef struct {
+ uint8_t fast_period_divisor; /*!< Divider for the Publish Period */
+} esp_ble_mesh_health_period_status_cb_t;
+
+typedef struct {
+ uint8_t attention; /*!< Value of the Attention Timer state */
+} esp_ble_mesh_health_attention_status_cb_t;
+
+typedef union {
+ esp_ble_mesh_health_current_status_cb_t current_status; /*!< The health current status value */
+ esp_ble_mesh_health_fault_status_cb_t fault_status; /*!< The health fault status value */
+ esp_ble_mesh_health_period_status_cb_t period_status; /*!< The health period status value */
+ esp_ble_mesh_health_attention_status_cb_t attention_status; /*!< The health attention status value */
+} esp_ble_mesh_health_client_common_cb_param_t;
+
+typedef struct {
+ int error_code; /*!< Appropriate error code */
+ esp_ble_mesh_client_common_param_t *params; /*!< The client common parameters. */
+ esp_ble_mesh_health_client_common_cb_param_t status_cb; /*!< The health message status callback values */
+} esp_ble_mesh_health_client_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_HEALTH_CLIENT_GET_STATE_EVT,
+ ESP_BLE_MESH_HEALTH_CLIENT_SET_STATE_EVT,
+ ESP_BLE_MESH_HEALTH_CLIENT_PUBLISH_EVT,
+ ESP_BLE_MESH_HEALTH_CLIENT_TIMEOUT_EVT,
+ ESP_BLE_MESH_HEALTH_CLIENT_EVT_MAX,
+} esp_ble_mesh_health_client_cb_event_t;
+
+typedef struct {
+ int error_code; /*!< Appropriate error code */
+} esp_ble_mesh_health_server_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_HEALTH_SERVER_FAULT_UPDATE_COMPLETE_EVT,
+ ESP_BLE_MESH_HEALTH_SERVER_EVT_MAX,
+} esp_ble_mesh_health_server_cb_event_t;
+
+/**
+ * @brief Bluetooth Mesh Health Client and Server Model function.
+ */
+
+/** @brief: event, event code of Health Client Model event; param, parameters of Health Client Model event) */
+typedef void (* esp_ble_mesh_health_client_cb_t)(esp_ble_mesh_health_client_cb_event_t event,
+ esp_ble_mesh_health_client_cb_param_t *param);
+
+/** @brief: event, event code of Health Server Model event; param, parameters of Health Server Model event) */
+typedef void (* esp_ble_mesh_health_server_cb_t)(esp_ble_mesh_health_server_cb_event_t event,
+ esp_ble_mesh_health_server_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh Health Model callback, the callback will report Health Client & Server Model events.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_health_client_callback(esp_ble_mesh_health_client_cb_t callback);
+
+/**
+ * @brief Register BLE Mesh Health Server Model callback.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_health_server_callback(esp_ble_mesh_health_server_cb_t callback);
+
+/**
+ * @brief This function is called to get the Health Server states using the Health Client Model get messages.
+ *
+ * @note If you want to find the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_opcode_health_client_get_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] get_state: Pointer to a union, each kind of opcode corresponds to one structure inside.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_health_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_get_state_t *get_state);
+
+/**
+ * @brief This function is called to set the Health Server states using the Health Client Model set messages.
+ *
+ * @note If you want to find the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_opcode_health_client_set_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] set_state: Pointer to a union, each kind of opcode corresponds to one structure inside.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_health_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_set_state_t *set_state);
+
+/**
+ * @brief This function is called by the Health Server Model to start to publish its Current Health Fault.
+ *
+ * @param[in] element: The element to which the Health Server Model belongs.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_health_server_fault_update(esp_ble_mesh_elem_t *element);
+
+#endif /** _ESP_BLE_MESH_HEALTH_MODEL_API_H_ */
diff --git a/components/bt/ble_mesh/api/models/include/esp_ble_mesh_lighting_model_api.h b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_lighting_model_api.h
new file mode 100644
index 0000000000..e0f849e8f6
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_lighting_model_api.h
@@ -0,0 +1,526 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Light Client Model APIs.
+ */
+
+#ifndef _ESP_BLE_MESH_LIGHTING_MODEL_API_H_
+#define _ESP_BLE_MESH_LIGHTING_MODEL_API_H_
+
+#include "light_client.h"
+#include "esp_ble_mesh_defs.h"
+
+/** @def ESP_BLE_MESH_MODEL_LIGHT_LIGHTNESS_CLI
+ *
+ * @brief Define a new Light Lightness Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Light Lightness Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Light Lightness Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_LIGHT_LIGHTNESS_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_LIGHT_LIGHTNESS_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_LIGHT_CTL_CLI
+ *
+ * @brief Define a new Light CTL Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Light CTL Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Light CTL Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_LIGHT_CTL_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_LIGHT_CTL_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_LIGHT_HSL_CLI
+ *
+ * @brief Define a new Light HSL Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Light HSL Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Light HSL Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_LIGHT_HSL_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_LIGHT_HSL_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_LIGHT_XYL_CLI
+ *
+ * @brief Define a new Light xyL Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Light xyL Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Light xyL Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_LIGHT_XYL_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_LIGHT_XYL_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_LIGHT_LC_CLI
+ *
+ * @brief Define a new Light LC Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Light LC Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Light LC Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_LIGHT_LC_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_LIGHT_LC_CLI, \
+ NULL, cli_pub, cli_data)
+
+/**
+ * @brief Bluetooth Mesh Light Lightness Client Model Get and Set parameters structure.
+ */
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t lightness; /* Target value of light lightness actual state */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_lightness_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t lightness; /* Target value of light lightness linear state */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_lightness_linear_set_t;
+
+typedef struct {
+ u16_t lightness; /* The value of the Light Lightness Default state */
+} esp_ble_mesh_light_lightness_default_set_t;
+
+typedef struct {
+ u16_t range_min; /* Value of range min field of light lightness range state */
+ u16_t range_max; /* Value of range max field of light lightness range state */
+} esp_ble_mesh_light_lightness_range_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t ctl_lightness; /* Target value of light ctl lightness state */
+ u16_t ctl_temperatrue; /* Target value of light ctl temperature state */
+ s16_t ctl_delta_uv; /* Target value of light ctl delta UV state */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_ctl_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t ctl_temperatrue; /* Target value of light ctl temperature state */
+ s16_t ctl_delta_uv; /* Target value of light ctl delta UV state */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_ctl_temperature_set_t;
+
+typedef struct {
+ u16_t range_min; /* Value of temperature range min field of light ctl temperature range state */
+ u16_t range_max; /* Value of temperature range max field of light ctl temperature range state */
+} esp_ble_mesh_light_ctl_temperature_range_set_t;
+
+typedef struct {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t temperature; /* Value of light temperature default state */
+ s16_t delta_uv; /* Value of light delta UV default state */
+} esp_ble_mesh_light_ctl_default_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t hsl_lightness; /* Target value of light hsl lightness state */
+ u16_t hsl_hue; /* Target value of light hsl hue state */
+ u16_t hsl_saturation; /* Target value of light hsl saturation state */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_hsl_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t hue; /* Target value of light hsl hue state */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_hsl_hue_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t saturation; /* Target value of light hsl hue state */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_hsl_saturation_set_t;
+
+typedef struct {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t hue; /* Value of light hue default state */
+ u16_t saturation; /* Value of light saturation default state */
+} esp_ble_mesh_light_hsl_default_set_t;
+
+typedef struct {
+ u16_t hue_range_min; /* Value of hue range min field of light hsl hue range state */
+ u16_t hue_range_max; /* Value of hue range max field of light hsl hue range state */
+ u16_t saturation_range_min; /* Value of saturation range min field of light hsl saturation range state */
+ u16_t saturation_range_max; /* Value of saturation range max field of light hsl saturation range state */
+} esp_ble_mesh_light_hsl_range_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t xyl_lightness; /* The target value of the Light xyL Lightness state */
+ u16_t xyl_x; /* The target value of the Light xyL x state */
+ u16_t xyl_y; /* The target value of the Light xyL y state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_xyl_set_t;
+
+typedef struct {
+ u16_t lightness; /* The value of the Light Lightness Default state */
+ u16_t xyl_x; /* The value of the Light xyL x Default state */
+ u16_t xyl_y; /* The value of the Light xyL y Default state */
+} esp_ble_mesh_light_xyl_default_set_t;
+
+typedef struct {
+ u16_t xyl_x_range_min; /* The value of the xyL x Range Min field of the Light xyL x Range state */
+ u16_t xyl_x_range_max; /* The value of the xyL x Range Max field of the Light xyL x Range state */
+ u16_t xyl_y_range_min; /* The value of the xyL y Range Min field of the Light xyL y Range state */
+ u16_t xyl_y_range_max; /* The value of the xyL y Range Max field of the Light xyL y Range state */
+} esp_ble_mesh_light_xyl_range_set_t;
+
+typedef struct {
+ u8_t mode; /* The target value of the Light LC Mode state */
+} esp_ble_mesh_light_lc_mode_set_t;
+
+typedef struct {
+ u8_t mode; /* The target value of the Light LC Occupancy Mode state */
+} esp_ble_mesh_light_lc_om_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate whether optional parameters included */
+ u8_t light_onoff; /* The target value of the Light LC Light OnOff state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_light_lc_light_onoff_set_t;
+
+typedef struct {
+ u16_t property_id; /* Property ID identifying a Light LC Property */
+} esp_ble_mesh_light_lc_property_get_t;
+
+typedef struct {
+ u16_t property_id; /* Property ID identifying a Light LC Property */
+ struct net_buf_simple *property_value; /* Raw value for the Light LC Property */
+} esp_ble_mesh_light_lc_property_set_t;
+
+typedef union {
+ esp_ble_mesh_light_lc_property_get_t lc_property_get; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET */
+} esp_ble_mesh_light_client_get_state_t;
+
+typedef union {
+ esp_ble_mesh_light_lightness_set_t lightness_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK */
+ esp_ble_mesh_light_lightness_linear_set_t lightness_linear_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK */
+ esp_ble_mesh_light_lightness_default_set_t lightness_default_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK */
+ esp_ble_mesh_light_lightness_range_set_t lightness_range_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK */
+ esp_ble_mesh_light_ctl_set_t ctl_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK */
+ esp_ble_mesh_light_ctl_temperature_set_t ctl_temperature_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK */
+ esp_ble_mesh_light_ctl_temperature_range_set_t ctl_temperature_range_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK */
+ esp_ble_mesh_light_ctl_default_set_t ctl_default_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK */
+ esp_ble_mesh_light_hsl_set_t hsl_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK */
+ esp_ble_mesh_light_hsl_hue_set_t hsl_hue_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK */
+ esp_ble_mesh_light_hsl_saturation_set_t hsl_saturation_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK */
+ esp_ble_mesh_light_hsl_default_set_t hsl_default_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK */
+ esp_ble_mesh_light_hsl_range_set_t hsl_range_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK */
+ esp_ble_mesh_light_xyl_set_t xyl_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK */
+ esp_ble_mesh_light_xyl_default_set_t xyl_default_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK */
+ esp_ble_mesh_light_xyl_range_set_t xyl_range_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK */
+ esp_ble_mesh_light_lc_mode_set_t lc_mode_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK */
+ esp_ble_mesh_light_lc_om_set_t lc_om_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK */
+ esp_ble_mesh_light_lc_light_onoff_set_t lc_light_onoff_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK */
+ esp_ble_mesh_light_lc_property_set_t lc_property_set; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET & ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK */
+} esp_ble_mesh_light_client_set_state_t;
+
+/**
+ * @brief Bluetooth Mesh Light Lightness Client Model Get and Set callback parameters structure.
+ */
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t present_lightness; /* Current value of light lightness actual state */
+ u16_t target_lightness; /* Target value of light lightness actual state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_light_lightness_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t present_lightness; /* Current value of light lightness linear state */
+ u16_t target_lightness; /* Target value of light lightness linear state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_light_lightness_linear_status_cb_t;
+
+typedef struct {
+ u16_t lightness; /* The value of the Light Lightness Last state */
+} esp_ble_mesh_light_lightness_last_status_cb_t;
+
+typedef struct {
+ u16_t lightness; /* The value of the Light Lightness default State */
+} esp_ble_mesh_light_lightness_default_status_cb_t;
+
+typedef struct {
+ u8_t status_code; /* Status Code for the request message */
+ u16_t range_min; /* Value of range min field of light lightness range state */
+ u16_t range_max; /* Value of range max field of light lightness range state */
+} esp_ble_mesh_light_lightness_range_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t present_ctl_lightness; /* Current value of light ctl lightness state */
+ u16_t present_ctl_temperature; /* Current value of light ctl temperature state */
+ u16_t target_ctl_lightness; /* Target value of light ctl lightness state (optional) */
+ u16_t target_ctl_temperature; /* Target value of light ctl temperature state (C.1) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_light_ctl_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t present_ctl_temperature; /* Current value of light ctl temperature state */
+ u16_t present_ctl_delta_uv; /* Current value of light ctl delta UV state */
+ u16_t target_ctl_temperature; /* Target value of light ctl temperature state (optional) */
+ u16_t target_ctl_delta_uv; /* Target value of light ctl delta UV state (C.1) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_light_ctl_temperature_status_cb_t;
+
+typedef struct {
+ u8_t status_code; /* Status code for the request message */
+ u16_t range_min; /* Value of temperature range min field of light ctl temperature range state */
+ u16_t range_max; /* Value of temperature range max field of light ctl temperature range state */
+} esp_ble_mesh_light_ctl_temperature_range_status_cb_t;
+
+typedef struct {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t temperature; /* Value of light temperature default state */
+ s16_t delta_uv; /* Value of light delta UV default state */
+} esp_ble_mesh_light_ctl_default_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t hsl_lightness; /* Current value of light hsl lightness state */
+ u16_t hsl_hue; /* Current value of light hsl hue state */
+ u16_t hsl_saturation; /* Current value of light hsl saturation state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+} esp_ble_mesh_light_hsl_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t hsl_lightness_target; /* Target value of light hsl lightness state */
+ u16_t hsl_hue_target; /* Target value of light hsl hue state */
+ u16_t hsl_saturation_target; /* Target value of light hsl saturation state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+} esp_ble_mesh_light_hsl_target_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t present_hue; /* Current value of light hsl hue state */
+ u16_t target_hue; /* Target value of light hsl hue state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_light_hsl_hue_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t present_saturation; /* Current value of light hsl saturation state */
+ u16_t target_saturation; /* Target value of light hsl saturation state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_light_hsl_saturation_status_cb_t;
+
+typedef struct {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t hue; /* Value of light hue default state */
+ u16_t saturation; /* Value of light saturation default state */
+} esp_ble_mesh_light_hsl_default_status_cb_t;
+
+typedef struct {
+ u8_t status_code; /* Status code for the request message */
+ u16_t hue_range_min; /* Value of hue range min field of light hsl hue range state */
+ u16_t hue_range_max; /* Value of hue range max field of light hsl hue range state */
+ u16_t saturation_range_min; /* Value of saturation range min field of light hsl saturation range state */
+ u16_t saturation_range_max; /* Value of saturation range max field of light hsl saturation range state */
+} esp_ble_mesh_light_hsl_range_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t xyl_lightness; /* The present value of the Light xyL Lightness state */
+ u16_t xyl_x; /* The present value of the Light xyL x state */
+ u16_t xyl_y; /* The present value of the Light xyL y state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+} esp_ble_mesh_light_xyl_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t target_xyl_lightness; /* The target value of the Light xyL Lightness state */
+ u16_t target_xyl_x; /* The target value of the Light xyL x state */
+ u16_t target_xyl_y; /* The target value of the Light xyL y state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+} esp_ble_mesh_light_xyl_target_status_cb_t;
+
+typedef struct {
+ u16_t lightness; /* The value of the Light Lightness Default state */
+ u16_t xyl_x; /* The value of the Light xyL x Default state */
+ u16_t xyl_y; /* The value of the Light xyL y Default state */
+} esp_ble_mesh_light_xyl_default_status_cb_t;
+
+typedef struct {
+ u8_t status_code; /* Status Code for the requesting message */
+ u16_t xyl_x_range_min; /* The value of the xyL x Range Min field of the Light xyL x Range state */
+ u16_t xyl_x_range_max; /* The value of the xyL x Range Max field of the Light xyL x Range state */
+ u16_t xyl_y_range_min; /* The value of the xyL y Range Min field of the Light xyL y Range state */
+ u16_t xyl_y_range_max; /* The value of the xyL y Range Max field of the Light xyL y Range state */
+} esp_ble_mesh_light_xyl_range_status_cb_t;
+
+typedef struct {
+ u8_t mode; /* The present value of the Light LC Mode state */
+} esp_ble_mesh_light_lc_mode_status_cb_t;
+
+typedef struct {
+ u8_t mode; /* The present value of the Light LC Occupancy Mode state */
+} esp_ble_mesh_light_lc_om_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate whether optional parameters included */
+ u8_t present_light_onoff; /* The present value of the Light LC Light OnOff state */
+ u8_t target_light_onoff; /* The target value of the Light LC Light OnOff state (Optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_light_lc_light_onoff_status_cb_t;
+
+typedef struct {
+ u16_t property_id; /* Property ID identifying a Light LC Property */
+ struct net_buf_simple *property_value; /* Raw value for the Light LC Property */
+} esp_ble_mesh_light_lc_property_status_cb_t;
+
+typedef union {
+ esp_ble_mesh_light_lightness_status_cb_t lightness_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS */
+ esp_ble_mesh_light_lightness_linear_status_cb_t lightness_linear_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS */
+ esp_ble_mesh_light_lightness_last_status_cb_t lightness_last_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS */
+ esp_ble_mesh_light_lightness_default_status_cb_t lightness_default_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS */
+ esp_ble_mesh_light_lightness_range_status_cb_t lightness_range_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS */
+ esp_ble_mesh_light_ctl_status_cb_t ctl_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_STATUS */
+ esp_ble_mesh_light_ctl_temperature_status_cb_t ctl_temperature_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS */
+ esp_ble_mesh_light_ctl_temperature_range_status_cb_t ctl_temperature_range_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS */
+ esp_ble_mesh_light_ctl_default_status_cb_t ctl_default_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS */
+ esp_ble_mesh_light_hsl_status_cb_t hsl_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_STATUS */
+ esp_ble_mesh_light_hsl_target_status_cb_t hsl_target_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS */
+ esp_ble_mesh_light_hsl_hue_status_cb_t hsl_hue_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS */
+ esp_ble_mesh_light_hsl_saturation_status_cb_t hsl_saturation_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS */
+ esp_ble_mesh_light_hsl_default_status_cb_t hsl_default_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS */
+ esp_ble_mesh_light_hsl_range_status_cb_t hsl_range_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS */
+ esp_ble_mesh_light_xyl_status_cb_t xyl_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_STATUS */
+ esp_ble_mesh_light_xyl_target_status_cb_t xyl_target_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS */
+ esp_ble_mesh_light_xyl_default_status_cb_t xyl_default_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS */
+ esp_ble_mesh_light_xyl_range_status_cb_t xyl_range_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS */
+ esp_ble_mesh_light_lc_mode_status_cb_t lc_mode_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS */
+ esp_ble_mesh_light_lc_om_status_cb_t lc_om_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_OM_STATUS */
+ esp_ble_mesh_light_lc_light_onoff_status_cb_t lc_light_onoff_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS */
+ esp_ble_mesh_light_lc_property_status_cb_t lc_property_status; /*!< For ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS */
+} esp_ble_mesh_light_client_status_cb_t;
+
+typedef struct {
+ int error_code; /*!< Appropriate error code */
+ esp_ble_mesh_client_common_param_t *params; /*!< The client common parameters. */
+ esp_ble_mesh_light_client_status_cb_t status_cb; /*!< The light status message callback values */
+} esp_ble_mesh_light_client_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_LIGHT_CLIENT_GET_STATE_EVT,
+ ESP_BLE_MESH_LIGHT_CLIENT_SET_STATE_EVT,
+ ESP_BLE_MESH_LIGHT_CLIENT_PUBLISH_EVT,
+ ESP_BLE_MESH_LIGHT_CLIENT_TIMEOUT_EVT,
+ ESP_BLE_MESH_LIGHT_CLIENT_EVT_MAX,
+} esp_ble_mesh_light_client_cb_event_t;
+
+/**
+ * @brief Bluetooth Mesh Light Client Model function.
+ */
+
+/** @brief: event, event code of Light Client Model events; param, parameters of Light Client Model events */
+typedef void (* esp_ble_mesh_light_client_cb_t)(esp_ble_mesh_light_client_cb_event_t event,
+ esp_ble_mesh_light_client_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh Light Client Model callback.
+ *
+ * @param[in] callback: pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_light_client_callback(esp_ble_mesh_light_client_cb_t callback);
+
+/**
+ * @brief Get the value of Light Server Model states using the Light Client Model get messages.
+ *
+ * @note If you want to know the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_light_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] get_state: Pointer of light get message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_light_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_light_client_get_state_t *get_state);
+
+/**
+ * @brief Set the value of Light Server Model states using the Light Client Model set messages.
+ *
+ * @note If you want to know the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_light_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] set_state: Pointer of generic set message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_light_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_light_client_set_state_t *set_state);
+
+
+#endif /* _ESP_BLE_MESH_LIGHTING_MODEL_API_H_ */
+
diff --git a/components/bt/ble_mesh/api/models/include/esp_ble_mesh_sensor_model_api.h b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_sensor_model_api.h
new file mode 100644
index 0000000000..553845cad8
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_sensor_model_api.h
@@ -0,0 +1,230 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Sensor Client Model APIs.
+ */
+
+#ifndef _ESP_BLE_MESH_SENSOR_MODEL_API_H_
+#define _ESP_BLE_MESH_SENSOR_MODEL_API_H_
+
+#include "sensor_client.h"
+#include "esp_ble_mesh_defs.h"
+
+/** @def ESP_BLE_MESH_MODEL_SENSOR_CLI
+ *
+ * @brief Define a new Sensor Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Sensor Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Sensor Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_SENSOR_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_SENSOR_CLI, \
+ NULL, cli_pub, cli_data)
+
+/**
+ * @brief Bluetooth Mesh Sensor Client Model Get and Set parameters structure.
+ */
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t property_id; /* Property ID of a sensor (optional) */
+} esp_ble_mesh_sensor_descriptor_get_t;
+
+typedef struct {
+ u16_t property_id; /* Property ID of a sensor */
+} esp_ble_mesh_sensor_cadence_get_t;
+
+typedef struct {
+ u16_t property_id; /* Property ID for the sensor */
+ u8_t fast_cadence_period_divisor : 7, /* Divisor for the publish period */
+ status_trigger_type : 1; /* The unit and format of the Status Trigger Delta fields */
+ struct net_buf_simple *status_trigger_delta_down; /* Delta down value that triggers a status message */
+ struct net_buf_simple *status_trigger_delta_up; /* Delta up value that triggers a status message */
+ u8_t status_min_interval; /* Minimum interval between two consecutive Status messages */
+ struct net_buf_simple *fast_cadence_low; /* Low value for the fast cadence range */
+ struct net_buf_simple *fast_cadence_high; /* Fast value for the fast cadence range */
+} esp_ble_mesh_sensor_cadence_set_t;
+
+typedef struct {
+ u16_t sensor_property_id; /* Property ID of a sensor */
+} esp_ble_mesh_sensor_settings_get_t;
+
+typedef struct {
+ u16_t sensor_property_id; /* Property ID of a sensor */
+ u16_t sensor_setting_property_id; /* Setting ID identifying a setting within a sensor */
+} esp_ble_mesh_sensor_setting_get_t;
+
+typedef struct {
+ u16_t sensor_property_id; /* Property ID identifying a sensor */
+ u16_t sensor_setting_property_id; /* Setting ID identifying a setting within a sensor */
+ struct net_buf_simple *sensor_setting_raw; /* Raw value for the setting */
+} esp_ble_mesh_sensor_setting_set_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t property_id; /* Property ID for the sensor (optional) */
+} esp_ble_mesh_sensor_get_t;
+
+typedef struct {
+ u16_t property_id; /* Property identifying a sensor */
+ struct net_buf_simple *raw_value_x; /* Raw value identifying a column */
+} esp_ble_mesh_sensor_column_get_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t property_id; /* Property identifying a sensor */
+ struct net_buf_simple *raw_value_x1; /* Raw value identifying a starting column (optional) */
+ struct net_buf_simple *raw_value_x2; /* Raw value identifying an ending column (C.1) */
+} esp_ble_mesh_sensor_series_get_t;
+
+typedef union {
+ esp_ble_mesh_sensor_descriptor_get_t descriptor_get; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET */
+ esp_ble_mesh_sensor_cadence_get_t cadence_get; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_GET */
+ esp_ble_mesh_sensor_settings_get_t settings_get; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_GET */
+ esp_ble_mesh_sensor_setting_get_t setting_get; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_GET */
+ esp_ble_mesh_sensor_get_t sensor_get; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_GET */
+ esp_ble_mesh_sensor_column_get_t column_get; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_GET */
+ esp_ble_mesh_sensor_series_get_t series_get; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_GET */
+} esp_ble_mesh_sensor_client_get_state_t;
+
+typedef union {
+ esp_ble_mesh_sensor_cadence_set_t cadence_set; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET & ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET_UNACK */
+ esp_ble_mesh_sensor_setting_set_t setting_set; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET & ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET_UNACK */
+} esp_ble_mesh_sensor_client_set_state_t;
+
+/**
+ * @brief Bluetooth Mesh Sensor Client Model Get and Set callback parameters structure.
+ */
+
+typedef struct {
+ struct net_buf_simple *descriptor; /* Sequence of 8-octet sensor descriptors (optional) */
+} esp_ble_mesh_sensor_descriptor_status_cb_t;
+
+typedef struct {
+ u16_t property_id; /* Property for the sensor */
+ struct net_buf_simple *sensor_cadence_value; /* Value of sensor cadence state */
+} esp_ble_mesh_sensor_cadence_status_cb_t;
+
+typedef struct {
+ u16_t sensor_property_id; /* Property ID identifying a sensor */
+ struct net_buf_simple *sensor_setting_property_ids; /* A sequence of N sensor setting property IDs (optional) */
+} esp_ble_mesh_sensor_settings_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate id optional parameters are included */
+ u16_t sensor_property_id; /* Property ID identifying a sensor */
+ u16_t sensor_setting_property_id; /* Setting ID identifying a setting within a sensor */
+ u8_t sensor_setting_access; /* Read/Write access rights for the setting (optional) */
+ struct net_buf_simple *sensor_setting_raw; /* Raw value for the setting */
+} esp_ble_mesh_sensor_setting_status_cb_t;
+
+typedef struct {
+ struct net_buf_simple *marshalled_sensor_data; /* Value of sensor data state (optional) */
+} esp_ble_mesh_sensor_status_cb_t;
+
+typedef struct {
+ u16_t property_id; /* Property identifying a sensor and the Y axis */
+ struct net_buf_simple *sensor_column_value; /* Left values of sensor column status */
+} esp_ble_mesh_sensor_column_status_cb_t;
+
+typedef struct {
+ u16_t property_id; /* Property identifying a sensor and the Y axis */
+ struct net_buf_simple *sensor_series_value; /* Left values of sensor series status */
+} esp_ble_mesh_sensor_series_status_cb_t;
+
+
+typedef union {
+ esp_ble_mesh_sensor_descriptor_status_cb_t descriptor_status; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS */
+ esp_ble_mesh_sensor_cadence_status_cb_t cadence_status; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_STATUS */
+ esp_ble_mesh_sensor_settings_status_cb_t settings_status; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS */
+ esp_ble_mesh_sensor_setting_status_cb_t setting_status; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_STATUS */
+ esp_ble_mesh_sensor_status_cb_t sensor_status; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_STATUS */
+ esp_ble_mesh_sensor_column_status_cb_t column_status; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_STATUS */
+ esp_ble_mesh_sensor_series_status_cb_t series_status; /*!< For ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_STATUS */
+} esp_ble_mesh_sensor_client_status_cb_t;
+
+typedef struct {
+ int error_code; /*!< 0: success,
+ * otherwise failure. For the error code values please refer to errno.h file.
+ * A negative sign is added to the standard error codes in errno.h. */
+ esp_ble_mesh_client_common_param_t *params; /*!< The client common parameters. */
+ esp_ble_mesh_sensor_client_status_cb_t status_cb; /*!< The sensor status message callback values */
+} esp_ble_mesh_sensor_client_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_SENSOR_CLIENT_GET_STATE_EVT,
+ ESP_BLE_MESH_SENSOR_CLIENT_SET_STATE_EVT,
+ ESP_BLE_MESH_SENSOR_CLIENT_PUBLISH_EVT,
+ ESP_BLE_MESH_SENSOR_CLIENT_TIMEOUT_EVT,
+ ESP_BLE_MESH_SENSOR_CLIENT_EVT_MAX,
+} esp_ble_mesh_sensor_client_cb_event_t;
+
+/**
+ * @brief Bluetooth Mesh Sensor Client Model function.
+ */
+
+/** @brief: event, event code of Sensor Client Model events; param, parameters of Sensor Client Model events */
+typedef void (* esp_ble_mesh_sensor_client_cb_t)(esp_ble_mesh_sensor_client_cb_event_t event,
+ esp_ble_mesh_sensor_client_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh Sensor Client Model callback.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_sensor_client_callback(esp_ble_mesh_sensor_client_cb_t callback);
+
+/**
+ * @brief Get the value of Sensor Server Model states using the Sensor Client Model get messages.
+ *
+ * @note If you want to know the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_sensor_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] get_state: Pointer to sensor get message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_sensor_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_sensor_client_get_state_t *get_state);
+
+/**
+ * @brief Set the value of Sensor Server Model states using the Sensor Client Model set messages.
+ *
+ * @note If you want to know the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_sensor_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] set_state: Pointer to sensor set message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_sensor_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_sensor_client_set_state_t *set_state);
+
+#endif /* _ESP_BLE_MESH_SENSOR_MODEL_API_H_ */
+
+
diff --git a/components/bt/ble_mesh/api/models/include/esp_ble_mesh_time_scene_model_api.h b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_time_scene_model_api.h
new file mode 100644
index 0000000000..cdf55ef170
--- /dev/null
+++ b/components/bt/ble_mesh/api/models/include/esp_ble_mesh_time_scene_model_api.h
@@ -0,0 +1,292 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Time and Scene Client Model APIs.
+ */
+
+#ifndef _ESP_BLE_MESH_TIME_SCENE_MODEL_API_H_
+#define _ESP_BLE_MESH_TIME_SCENE_MODEL_API_H_
+
+#include "time_scene_client.h"
+#include "esp_ble_mesh_defs.h"
+
+/** @def ESP_BLE_MESH_MODEL_TIME_CLI
+ *
+ * @brief Define a new Time Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Time Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Time Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_TIME_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_TIME_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_SCENE_CLI
+ *
+ * @brief Define a new Scene Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Scene Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Scene Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_SCENE_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_SCENE_CLI, \
+ NULL, cli_pub, cli_data)
+
+/** @def ESP_BLE_MESH_MODEL_SCHEDULER_CLI
+ *
+ * @brief Define a new Scheduler Client Model.
+ *
+ * @note This API needs to be called for each element on which
+ * the application needs to have a Scheduler Client Model.
+ *
+ * @param cli_pub Pointer to the unique struct esp_ble_mesh_model_pub_t.
+ * @param cli_data Pointer to the unique struct esp_ble_mesh_client_t.
+ *
+ * @return New Scheduler Client Model instance.
+ */
+#define ESP_BLE_MESH_MODEL_SCHEDULER_CLI(cli_pub, cli_data) \
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_SCHEDULER_CLI, \
+ NULL, cli_pub, cli_data)
+
+/**
+ * @brief Bluetooth Mesh Time Scene Client Model Get and Set parameters structure.
+ */
+
+typedef struct {
+ u8_t tai_seconds[5]; /* The current TAI time in seconds */
+ u8_t sub_second; /* The sub-second time in units of 1/256 second */
+ u8_t uncertainty; /* The estimated uncertainty in 10-millisecond steps */
+ u16_t time_authority : 1; /* 0 = No Time Authority, 1 = Time Authority */
+ u16_t tai_utc_delta : 15; /* Current difference between TAI and UTC in seconds */
+ u8_t time_zone_offset; /* The local time zone offset in 15-minute increments */
+} esp_ble_mesh_time_set_t;
+
+typedef struct {
+ u8_t time_zone_offset_new; /* Upcoming local time zone offset */
+ u8_t tai_zone_change[5]; /* TAI Seconds time of the upcoming Time Zone Offset change */
+} esp_ble_mesh_time_zone_set_t;
+
+typedef struct {
+ u16_t tai_utc_delta_new : 15; /* Upcoming difference between TAI and UTC in seconds */
+ u16_t padding : 1; /* Always 0b0. Other values are Prohibited. */
+ u8_t tai_delta_change[5]; /* TAI Seconds time of the upcoming TAI-UTC Delta change */
+} esp_ble_mesh_tai_utc_delta_set_t;
+
+typedef struct {
+ u8_t time_role; /* The Time Role for the element */
+} esp_ble_mesh_time_role_set_t;
+
+typedef struct {
+ u16_t scene_number; /* The number of scenes to be stored */
+} esp_ble_mesh_scene_store_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u16_t scene_number; /* The number of scenes to be recalled */
+ u8_t tid; /* Transaction ID */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+} esp_ble_mesh_scene_recall_t;
+
+typedef struct {
+ u16_t scene_number; /* The number of scenes to be deleted */
+} esp_ble_mesh_scene_delete_t;
+
+typedef struct {
+ u8_t index; /* Index of the Schedule Register entry to get */
+} esp_ble_mesh_scheduler_act_get_t;
+
+typedef struct {
+ u64_t index : 4; /* Index of the Schedule Register entry to set */
+ u64_t year : 7; /* Scheduled year for the action */
+ u64_t month : 12; /* Scheduled month for the action */
+ u64_t day : 5; /* Scheduled day of the month for the action */
+ u64_t hour : 5; /* Scheduled hour for the action */
+ u64_t minute : 6; /* Scheduled minute for the action */
+ u64_t second : 6; /* Scheduled second for the action */
+ u64_t day_of_week : 7; /* Schedule days of the week for the action */
+ u64_t action : 4; /* Action to be performed at the scheduled time */
+ u64_t trans_time : 8; /* Transition time for this action */
+ u16_t scene_number; /* Transition time for this action */
+} esp_ble_mesh_scheduler_act_set_t;
+
+/**
+ * @brief For
+ *
+ * the get_state parameter in the esp_ble_mesh_time_scene_client_get_state function should be set to NULL.
+ */
+typedef union {
+ esp_ble_mesh_scheduler_act_get_t scheduler_act_get; /*!< For ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_GET */
+} esp_ble_mesh_time_scene_client_get_state_t;
+
+typedef union {
+ esp_ble_mesh_time_set_t time_set; /*!< For ESP_BLE_MESH_MODEL_OP_TIME_SET */
+ esp_ble_mesh_time_zone_set_t time_zone_set; /*!< For ESP_BLE_MESH_MODEL_OP_TIME_ZONE_SET */
+ esp_ble_mesh_tai_utc_delta_set_t tai_utc_delta_set; /*!< For ESP_BLE_MESH_MODEL_OP_TAI_UTC_DELTA_SET */
+ esp_ble_mesh_time_role_set_t time_role_set; /*!< For ESP_BLE_MESH_MODEL_OP_TIME_ROLE_SET */
+ esp_ble_mesh_scene_store_t scene_store; /*!< For ESP_BLE_MESH_MODEL_OP_SCENE_STORE & ESP_BLE_MESH_MODEL_OP_SCENE_STORE_UNACK */
+ esp_ble_mesh_scene_recall_t scene_recall; /*!< For ESP_BLE_MESH_MODEL_OP_SCENE_RECALL & ESP_BLE_MESH_MODEL_OP_SCENE_RECALL_UNACK */
+ esp_ble_mesh_scene_delete_t scene_delete; /*!< For ESP_BLE_MESH_MODEL_OP_SCENE_DELETE & ESP_BLE_MESH_MODEL_OP_SCENE_DELETE_UNACK */
+ esp_ble_mesh_scheduler_act_set_t scheduler_act_set; /*!< For ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_SET & ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_SET_UNACK */
+} esp_ble_mesh_time_scene_client_set_state_t;
+
+/**
+ * @brief Bluetooth Mesh Time Scene Client Model Get and Set callback parameters structure.
+ */
+
+typedef struct {
+ u8_t tai_seconds[5]; /* The current TAI time in seconds */
+ u8_t sub_second; /* The sub-second time in units of 1/256 second */
+ u8_t uncertainty; /* The estimated uncertainty in 10-millisecond steps */
+ u16_t time_authority : 1; /* 0 = No Time Authority, 1 = Time Authority */
+ u16_t tai_utc_delta : 15; /* Current difference between TAI and UTC in seconds */
+ u8_t time_zone_offset; /* The local time zone offset in 15-minute increments */
+} esp_ble_mesh_time_status_cb_t;
+
+typedef struct {
+ u8_t time_zone_offset_curr; /* Current local time zone offset */
+ u8_t time_zone_offset_new; /* Upcoming local time zone offset */
+ u8_t tai_zone_change[5]; /* TAI Seconds time of the upcoming Time Zone Offset change */
+} esp_ble_mesh_time_zone_status_cb_t;
+
+typedef struct {
+ u16_t tai_utc_delta_curr : 15; /* Current difference between TAI and UTC in seconds */
+ u16_t padding_1 : 1; /* Always 0b0. Other values are Prohibited. */
+ u16_t tai_utc_delta_new : 15; /* Upcoming difference between TAI and UTC in seconds */
+ u16_t padding_2 : 1; /* Always 0b0. Other values are Prohibited. */
+ u8_t tai_delta_change[5]; /* TAI Seconds time of the upcoming TAI-UTC Delta change */
+} esp_ble_mesh_tai_utc_delta_status_cb_t;
+
+typedef struct {
+ u8_t time_role; /* The Time Role for the element */
+} esp_ble_mesh_time_role_status_cb_t;
+
+typedef struct {
+ bool op_en; /* Indicate if optional parameters are included */
+ u8_t status_code; /* Status code of the last operation */
+ u16_t current_scene; /* Scene Number of the current scene */
+ u16_t target_scene; /* Scene Number of the target scene (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+} esp_ble_mesh_scene_status_cb_t;
+
+typedef struct {
+ u8_t status_code; /* Status code for the previous operation */
+ u16_t current_scene; /* Scene Number of the current scene */
+ struct net_buf_simple *scenes; /* A list of scenes stored within an element */
+} esp_ble_mesh_scene_register_status_cb_t;
+
+typedef struct {
+ u16_t schedules; /* Bit field indicating defined Actions in the Schedule Register */
+} esp_ble_mesh_scheduler_status_cb_t;
+
+typedef struct {
+ u64_t index : 4; /* Enumerates (selects) a Schedule Register entry */
+ u64_t year : 7; /* Scheduled year for the action */
+ u64_t month : 12; /* Scheduled month for the action */
+ u64_t day : 5; /* Scheduled day of the month for the action */
+ u64_t hour : 5; /* Scheduled hour for the action */
+ u64_t minute : 6; /* Scheduled minute for the action */
+ u64_t second : 6; /* Scheduled second for the action */
+ u64_t day_of_week : 7; /* Schedule days of the week for the action */
+ u64_t action : 4; /* Action to be performed at the scheduled time */
+ u64_t trans_time : 8; /* Transition time for this action */
+ u16_t scene_number; /* Transition time for this action */
+} esp_ble_mesh_scheduler_act_status_cb_t;
+
+typedef union {
+ esp_ble_mesh_time_status_cb_t time_status; /*!< For ESP_BLE_MESH_MODEL_OP_TIME_STATUS */
+ esp_ble_mesh_time_zone_status_cb_t time_zone_status; /*!< For ESP_BLE_MESH_MODEL_OP_TIME_ZONE_STATUS */
+ esp_ble_mesh_tai_utc_delta_status_cb_t tai_utc_delta_status; /*!< For ESP_BLE_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS */
+ esp_ble_mesh_time_role_status_cb_t time_role_status; /*!< For ESP_BLE_MESH_MODEL_OP_TIME_ROLE_STATUS */
+ esp_ble_mesh_scene_status_cb_t scene_status; /*!< For ESP_BLE_MESH_MODEL_OP_SCENE_STATUS */
+ esp_ble_mesh_scene_register_status_cb_t scene_register_status; /*!< For ESP_BLE_MESH_MODEL_OP_SCENE_REGISTER_STATUS */
+ esp_ble_mesh_scheduler_status_cb_t scheduler_status; /*!< For ESP_BLE_MESH_MODEL_OP_SCHEDULER_STATUS */
+ esp_ble_mesh_scheduler_act_status_cb_t scheduler_act_status; /*!< For ESP_BLE_MESH_MODEL_OP_SCHEDULER_ACT_STATUS */
+} esp_ble_mesh_time_scene_client_status_cb_t;
+
+typedef struct {
+ int error_code; /*!< Appropriate error code */
+ esp_ble_mesh_client_common_param_t *params; /*!< The client common parameters. */
+ esp_ble_mesh_time_scene_client_status_cb_t status_cb; /*!< The scene status message callback values */
+} esp_ble_mesh_time_scene_client_cb_param_t;
+
+typedef enum {
+ ESP_BLE_MESH_TIME_SCENE_CLIENT_GET_STATE_EVT,
+ ESP_BLE_MESH_TIME_SCENE_CLIENT_SET_STATE_EVT,
+ ESP_BLE_MESH_TIME_SCENE_CLIENT_PUBLISH_EVT,
+ ESP_BLE_MESH_TIME_SCENE_CLIENT_TIMEOUT_EVT,
+ ESP_BLE_MESH_TIME_SCENE_CLIENT_EVT_MAX,
+} esp_ble_mesh_time_scene_client_cb_event_t;
+
+/**
+ * @brief Bluetooth Mesh Time Scene Client Model function.
+ */
+
+/** @brief: event, event code of Time Scene Client Model events; param, parameters of Time Scene Client Model events */
+typedef void (* esp_ble_mesh_time_scene_client_cb_t)(esp_ble_mesh_time_scene_client_cb_event_t event,
+ esp_ble_mesh_time_scene_client_cb_param_t *param);
+
+/**
+ * @brief Register BLE Mesh Time Scene Client Model callback.
+ *
+ * @param[in] callback: Pointer to the callback function.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ *
+ */
+esp_err_t esp_ble_mesh_register_time_scene_client_callback(esp_ble_mesh_time_scene_client_cb_t callback);
+
+/**
+ * @brief Get the value of Time Scene Server Model states using the Time Scene Client Model get messages.
+ *
+ * @note If you want to know the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_time_scene_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] get_state: Pointer to time scene get message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ */
+esp_err_t esp_ble_mesh_time_scene_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_time_scene_client_get_state_t *get_state);
+
+/**
+ * @brief Set the value of Time Scene Server Model states using the Time Scene Client Model set messages.
+ *
+ * @note If you want to know the opcodes and corresponding meanings accepted by this API,
+ * please refer to (@ref esp_ble_mesh_time_scene_message_opcode_t).
+ *
+ * @param[in] params: Pointer to BLE Mesh common client parameters.
+ * @param[in] set_state: Pointer to time scene set message value.
+ * Shall not be set to NULL.
+ *
+ * @return ESP_OK on success or error code otherwise.
+ */
+esp_err_t esp_ble_mesh_time_scene_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_time_scene_client_set_state_t *set_state);
+
+#endif /* _ESP_BLE_MESH_TIME_SCENE_MODEL_API_H_ */
+
diff --git a/components/bt/ble_mesh/btc/btc_ble_mesh_config_client.c b/components/bt/ble_mesh/btc/btc_ble_mesh_config_client.c
new file mode 100644
index 0000000000..3d82c2d725
--- /dev/null
+++ b/components/bt/ble_mesh/btc/btc_ble_mesh_config_client.c
@@ -0,0 +1,730 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "cfg_cli.h"
+#include "common.h"
+#include "btc_ble_mesh_config_client.h"
+#include "esp_ble_mesh_config_model_api.h"
+
+#define CID_NVAL 0xffff
+
+extern s32_t config_msg_timeout;
+
+static inline void btc_ble_mesh_cfg_client_cb_to_app(esp_ble_mesh_cfg_client_cb_event_t event,
+ esp_ble_mesh_cfg_client_cb_param_t *param)
+{
+ esp_ble_mesh_cfg_client_cb_t btc_mesh_cb = (esp_ble_mesh_cfg_client_cb_t)btc_profile_cb_get(BTC_PID_CFG_CLIENT);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+static inline void btc_ble_mesh_cfg_server_cb_to_app(esp_ble_mesh_cfg_server_cb_event_t event,
+ esp_ble_mesh_cfg_server_cb_param_t *param)
+{
+ esp_ble_mesh_cfg_server_cb_t btc_mesh_cb = (esp_ble_mesh_cfg_server_cb_t)btc_profile_cb_get(BTC_PID_CFG_SERVER);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+void btc_ble_mesh_cfg_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ btc_ble_mesh_cfg_client_args_t *dst = (btc_ble_mesh_cfg_client_args_t *)p_dest;
+ btc_ble_mesh_cfg_client_args_t *src = (btc_ble_mesh_cfg_client_args_t *)p_src;
+
+ if (!msg || !dst || !src) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_CONFIG_CLIENT_GET_STATE: {
+ dst->cfg_client_get_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->cfg_client_get_state.get_state = (esp_ble_mesh_cfg_client_get_state_t *)osi_malloc(sizeof(esp_ble_mesh_cfg_client_get_state_t));
+ if (dst->cfg_client_get_state.params && dst->cfg_client_get_state.get_state) {
+ memcpy(dst->cfg_client_get_state.params, src->cfg_client_get_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->cfg_client_get_state.get_state, src->cfg_client_get_state.get_state,
+ sizeof(esp_ble_mesh_cfg_client_get_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_CONFIG_CLIENT_SET_STATE: {
+ dst->cfg_client_set_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->cfg_client_set_state.set_state = (esp_ble_mesh_cfg_client_set_state_t *)osi_malloc(sizeof(esp_ble_mesh_cfg_client_set_state_t));
+ if (dst->cfg_client_set_state.params && dst->cfg_client_set_state.set_state) {
+ memcpy(dst->cfg_client_set_state.params, src->cfg_client_set_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->cfg_client_set_state.set_state, src->cfg_client_set_state.set_state,
+ sizeof(esp_ble_mesh_cfg_client_set_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ default:
+ LOG_DEBUG("%s Unhandled deep copy %d", __func__, msg->act);
+ break;
+ }
+}
+
+static void btc_ble_mesh_cfg_client_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ esp_ble_mesh_cfg_client_cb_param_t *p_dest_data = (esp_ble_mesh_cfg_client_cb_param_t *)p_dest;
+ esp_ble_mesh_cfg_client_cb_param_t *p_src_data = (esp_ble_mesh_cfg_client_cb_param_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !p_src_data || !p_dest_data) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_CFG_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_CFG_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_CFG_CLIENT_PUBLISH_EVT:
+ if (p_src_data->params) {
+ opcode = p_src_data->params->opcode;
+ switch (opcode) {
+ case OP_DEV_COMP_DATA_GET:
+ case OP_DEV_COMP_DATA_STATUS:
+ if (p_src_data->status_cb.comp_data_status.composition_data) {
+ length = p_src_data->status_cb.comp_data_status.composition_data->len;
+ p_dest_data->status_cb.comp_data_status.composition_data = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.comp_data_status.composition_data) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.comp_data_status.composition_data, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.comp_data_status.composition_data,
+ p_src_data->status_cb.comp_data_status.composition_data->data,
+ p_src_data->status_cb.comp_data_status.composition_data->len);
+ }
+ break;
+ case OP_MOD_SUB_GET:
+ case OP_MOD_SUB_GET_VND:
+ case OP_MOD_SUB_LIST:
+ case OP_MOD_SUB_LIST_VND:
+ if (p_src_data->status_cb.model_sub_list.sub_addr) {
+ length = p_src_data->status_cb.model_sub_list.sub_addr->len;
+ p_dest_data->status_cb.model_sub_list.sub_addr = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.model_sub_list.sub_addr) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.model_sub_list.sub_addr, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.model_sub_list.sub_addr,
+ p_src_data->status_cb.model_sub_list.sub_addr->data,
+ p_src_data->status_cb.model_sub_list.sub_addr->len);
+ }
+ break;
+ case OP_NET_KEY_GET:
+ case OP_NET_KEY_LIST:
+ if (p_src_data->status_cb.netkey_list.net_idx) {
+ length = p_src_data->status_cb.netkey_list.net_idx->len;
+ p_dest_data->status_cb.netkey_list.net_idx = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.netkey_list.net_idx) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.netkey_list.net_idx, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.netkey_list.net_idx,
+ p_src_data->status_cb.netkey_list.net_idx->data,
+ p_src_data->status_cb.netkey_list.net_idx->len);
+ }
+ break;
+ case OP_APP_KEY_GET:
+ case OP_APP_KEY_LIST:
+ if (p_src_data->status_cb.appkey_list.app_idx) {
+ length = p_src_data->status_cb.appkey_list.app_idx->len;
+ p_dest_data->status_cb.appkey_list.app_idx = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.appkey_list.app_idx) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.appkey_list.app_idx, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.appkey_list.app_idx,
+ p_src_data->status_cb.appkey_list.app_idx->data,
+ p_src_data->status_cb.appkey_list.app_idx->len);
+ }
+ break;
+ case OP_SIG_MOD_APP_GET:
+ case OP_VND_MOD_APP_GET:
+ case OP_SIG_MOD_APP_LIST:
+ case OP_VND_MOD_APP_LIST:
+ if (p_src_data->status_cb.model_app_list.app_idx) {
+ length = p_src_data->status_cb.model_app_list.app_idx->len;
+ p_dest_data->status_cb.model_app_list.app_idx = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.model_app_list.app_idx) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.model_app_list.app_idx, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.model_app_list.app_idx,
+ p_src_data->status_cb.model_app_list.app_idx->data,
+ p_src_data->status_cb.model_app_list.app_idx->len);
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_CFG_CLIENT_TIMEOUT_EVT:
+ if (p_src_data->params) {
+ p_dest_data->params = osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ if (p_dest_data->params) {
+ memcpy(p_dest_data->params, p_src_data->params, sizeof(esp_ble_mesh_client_common_param_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_cfg_client_free_req_data(btc_msg_t *msg)
+{
+ esp_ble_mesh_cfg_client_cb_param_t *arg = NULL;
+ u32_t opcode;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (esp_ble_mesh_cfg_client_cb_param_t *)(msg->arg);
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_CFG_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_CFG_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_CFG_CLIENT_PUBLISH_EVT:
+ if (arg->params) {
+ opcode = arg->params->opcode;
+ switch (opcode) {
+ case OP_DEV_COMP_DATA_GET:
+ case OP_DEV_COMP_DATA_STATUS:
+ bt_mesh_free_buf(arg->status_cb.comp_data_status.composition_data);
+ break;
+ case OP_MOD_SUB_GET:
+ case OP_MOD_SUB_GET_VND:
+ case OP_MOD_SUB_LIST:
+ case OP_MOD_SUB_LIST_VND:
+ bt_mesh_free_buf(arg->status_cb.model_sub_list.sub_addr);
+ break;
+ case OP_NET_KEY_GET:
+ case OP_NET_KEY_LIST:
+ bt_mesh_free_buf(arg->status_cb.netkey_list.net_idx);
+ break;
+ case OP_APP_KEY_GET:
+ case OP_APP_KEY_LIST:
+ bt_mesh_free_buf(arg->status_cb.appkey_list.app_idx);
+ break;
+ case OP_SIG_MOD_APP_GET:
+ case OP_VND_MOD_APP_GET:
+ case OP_SIG_MOD_APP_LIST:
+ case OP_VND_MOD_APP_LIST:
+ bt_mesh_free_buf(arg->status_cb.model_app_list.app_idx);
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_CFG_CLIENT_TIMEOUT_EVT:
+ if (arg->params) {
+ osi_free(arg->params);
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+void btc_ble_mesh_cfg_client_arg_deep_free(btc_msg_t *msg)
+{
+ btc_ble_mesh_cfg_client_args_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_cfg_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_CONFIG_CLIENT_GET_STATE:
+ if (arg->cfg_client_get_state.params) {
+ osi_free(arg->cfg_client_get_state.params);
+ }
+ if (arg->cfg_client_get_state.get_state) {
+ osi_free(arg->cfg_client_get_state.get_state);
+ }
+ break;
+ case BTC_BLE_MESH_ACT_CONFIG_CLIENT_SET_STATE:
+ if (arg->cfg_client_set_state.params) {
+ osi_free(arg->cfg_client_set_state.params);
+ }
+ if (arg->cfg_client_set_state.set_state) {
+ osi_free(arg->cfg_client_set_state.set_state);
+ }
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+static void btc_mesh_cfg_client_callback(esp_ble_mesh_cfg_client_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_CFG_CLIENT;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params,
+ sizeof(esp_ble_mesh_cfg_client_cb_param_t), btc_ble_mesh_cfg_client_copy_req_data);
+}
+
+void bt_mesh_callback_config_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len)
+{
+ esp_ble_mesh_cfg_client_cb_param_t cb_params = {0};
+ esp_ble_mesh_client_common_param_t params = {0};
+ size_t length;
+ uint8_t act;
+
+ if (!model || !ctx) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ switch (evt_type) {
+ case 0x00:
+ act = ESP_BLE_MESH_CFG_CLIENT_GET_STATE_EVT;
+ break;
+ case 0x01:
+ act = ESP_BLE_MESH_CFG_CLIENT_SET_STATE_EVT;
+ break;
+ case 0x02:
+ act = ESP_BLE_MESH_CFG_CLIENT_PUBLISH_EVT;
+ break;
+ case 0x03:
+ act = ESP_BLE_MESH_CFG_CLIENT_TIMEOUT_EVT;
+ break;
+ default:
+ LOG_ERROR("%s: unknown config status event type", __func__);
+ return;
+ }
+
+ params.opcode = opcode;
+ params.model = (esp_ble_mesh_model_t *)model;
+ params.ctx.net_idx = ctx->net_idx;
+ params.ctx.app_idx = ctx->app_idx;
+ params.ctx.addr = ctx->addr;
+ params.ctx.recv_ttl = ctx->recv_ttl;
+ params.ctx.recv_op = ctx->recv_op;
+ params.ctx.recv_dst = ctx->recv_dst;
+
+ cb_params.error_code = 0;
+ cb_params.params = ¶ms;
+
+ if (val && len) {
+ length = (len <= sizeof(cb_params.status_cb)) ? len : sizeof(cb_params.status_cb);
+ memcpy(&cb_params.status_cb, val, length);
+ }
+
+ btc_mesh_cfg_client_callback(&cb_params, act);
+}
+
+
+void btc_mesh_cfg_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf)
+{
+ if (!model || !ctx || !buf) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ bt_mesh_callback_config_status_to_btc(opcode, 0x02, model, ctx, buf->data, buf->len);
+}
+
+void btc_mesh_cfg_client_call_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_cfg_client_cb_param_t cfg_client_cb = {0};
+ btc_ble_mesh_cfg_client_args_t *arg = NULL;
+ bt_mesh_role_param_t role_param = {0};
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_cfg_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_CONFIG_CLIENT_GET_STATE: {
+ cfg_client_cb.params = arg->cfg_client_get_state.params;
+ role_param.model = (struct bt_mesh_model *)cfg_client_cb.params->model;
+ role_param.role = cfg_client_cb.params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ btc_ble_mesh_config_client_get_state(arg->cfg_client_get_state.params,
+ arg->cfg_client_get_state.get_state,
+ &cfg_client_cb);
+ if (cfg_client_cb.error_code) {
+ btc_mesh_cfg_client_callback(&cfg_client_cb, ESP_BLE_MESH_CFG_CLIENT_GET_STATE_EVT);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_CONFIG_CLIENT_SET_STATE: {
+ cfg_client_cb.params = arg->cfg_client_set_state.params;
+ role_param.model = (struct bt_mesh_model *)cfg_client_cb.params->model;
+ role_param.role = cfg_client_cb.params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ btc_ble_mesh_config_client_set_state(arg->cfg_client_set_state.params,
+ arg->cfg_client_set_state.set_state,
+ &cfg_client_cb);
+ if (cfg_client_cb.error_code) {
+ btc_mesh_cfg_client_callback(&cfg_client_cb, ESP_BLE_MESH_CFG_CLIENT_SET_STATE_EVT);
+ }
+ break;
+ }
+ default:
+ break;
+ }
+
+ btc_ble_mesh_cfg_client_arg_deep_free(msg);
+}
+
+void btc_mesh_cfg_client_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_cfg_client_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_cfg_client_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_CFG_CLIENT_EVT_MAX) {
+ btc_ble_mesh_cfg_client_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_cfg_client_free_req_data(msg);
+}
+
+int btc_ble_mesh_config_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_cfg_client_get_state_t *get_state,
+ esp_ble_mesh_cfg_client_cb_param_t *cfg_client_cb)
+{
+ struct bt_mesh_msg_ctx ctx = {0};
+
+ if (!params || !cfg_client_cb) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ ctx.net_idx = params->ctx.net_idx;
+ ctx.app_idx = BT_MESH_KEY_DEV;
+ ctx.addr = params->ctx.addr;
+ ctx.send_rel = params->ctx.send_rel;
+ ctx.send_ttl = params->ctx.send_ttl;
+
+ config_msg_timeout = params->msg_timeout;
+
+ switch (params->opcode) {
+ case ESP_BLE_MESH_MODEL_OP_BEACON_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_beacon_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_ttl_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_FRIEND_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_friend_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_GATT_PROXY_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_gatt_proxy_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_RELAY_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_relay_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_PUB_GET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_pub_get(&ctx, get_state->model_pub_get.element_addr, get_state->model_pub_get.model_id,
+ get_state->model_pub_get.company_id));
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_hb_pub_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_hb_sub_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_COMPOSITION_DATA_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_comp_data_get(&ctx, get_state->comp_data_get.page));
+ case ESP_BLE_MESH_MODEL_OP_SIG_MODEL_SUB_GET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_get(&ctx, get_state->sig_model_sub_get.element_addr, get_state->sig_model_sub_get.model_id));
+ case ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_SUB_GET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_get_vnd(&ctx, get_state->vnd_model_sub_get.element_addr,
+ get_state->vnd_model_sub_get.model_id, get_state->vnd_model_sub_get.company_id));
+ case ESP_BLE_MESH_MODEL_OP_NET_KEY_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_net_key_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_APP_KEY_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_app_key_get(&ctx, get_state->app_key_get.net_idx));
+ case ESP_BLE_MESH_MODEL_OP_NODE_IDENTITY_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_node_identity_get(&ctx, get_state->node_identity_get.net_idx));
+ case ESP_BLE_MESH_MODEL_OP_SIG_MODEL_APP_GET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_app_get(&ctx, get_state->sig_model_app_get.element_addr, get_state->sig_model_app_get.model_id));
+ case ESP_BLE_MESH_MODEL_OP_VENDOR_MODEL_APP_GET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_app_get_vnd(&ctx, get_state->vnd_model_app_get.element_addr,
+ get_state->vnd_model_app_get.model_id, get_state->vnd_model_app_get.company_id));
+ case ESP_BLE_MESH_MODEL_OP_KEY_REFRESH_PHASE_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_kr_phase_get(&ctx, get_state->kr_phase_get.net_idx));
+ case ESP_BLE_MESH_MODEL_OP_LPN_POLLTIMEOUT_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_lpn_timeout_get(&ctx, get_state->lpn_pollto_get.lpn_addr));
+ case ESP_BLE_MESH_MODEL_OP_NETWORK_TRANSMIT_GET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_net_transmit_get(&ctx));
+ default:
+ BT_WARN("%s, invalid opcode 0x%x", __func__, params->opcode);
+ return (cfg_client_cb->error_code = -EINVAL);
+ }
+
+ return 0;
+}
+
+int btc_ble_mesh_config_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_cfg_client_set_state_t *set_state,
+ esp_ble_mesh_cfg_client_cb_param_t *cfg_client_cb)
+{
+ struct bt_mesh_msg_ctx ctx = {0};
+
+ if (!params || !set_state || !cfg_client_cb) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ ctx.net_idx = params->ctx.net_idx;
+ ctx.app_idx = BT_MESH_KEY_DEV;
+ ctx.addr = params->ctx.addr;
+ ctx.send_rel = params->ctx.send_rel;
+ ctx.send_ttl = params->ctx.send_ttl;
+
+ config_msg_timeout = params->msg_timeout;
+
+ switch (params->opcode) {
+ case ESP_BLE_MESH_MODEL_OP_BEACON_SET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_beacon_set(&ctx, set_state->beacon_set.beacon));
+ case ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_SET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_ttl_set(&ctx, set_state->default_ttl_set.ttl));
+ case ESP_BLE_MESH_MODEL_OP_FRIEND_SET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_friend_set(&ctx, set_state->friend_set.friend_state));
+ case ESP_BLE_MESH_MODEL_OP_GATT_PROXY_SET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_gatt_proxy_set(&ctx, set_state->gatt_proxy_set.gatt_proxy));
+ case ESP_BLE_MESH_MODEL_OP_RELAY_SET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_relay_set(&ctx, set_state->relay_set.relay, set_state->relay_set.relay_retransmit));
+ case ESP_BLE_MESH_MODEL_OP_NET_KEY_ADD:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_net_key_add(&ctx, set_state->net_key_add.net_idx, &set_state->net_key_add.net_key[0]));
+ case ESP_BLE_MESH_MODEL_OP_APP_KEY_ADD:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_app_key_add(&ctx, set_state->app_key_add.net_idx,
+ set_state->app_key_add.app_idx, &set_state->app_key_add.app_key[0]));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_APP_BIND:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_app_bind(&ctx, set_state->model_app_bind.element_addr, set_state->model_app_bind.model_app_idx,
+ set_state->model_app_bind.model_id, set_state->model_app_bind.company_id));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_PUB_SET: {
+ struct bt_mesh_cfg_mod_pub model_pub = {
+ .addr = set_state->model_pub_set.publish_addr,
+ .app_idx = set_state->model_pub_set.publish_app_idx,
+ .cred_flag = set_state->model_pub_set.cred_flag,
+ .ttl = set_state->model_pub_set.publish_ttl,
+ .period = set_state->model_pub_set.publish_period,
+ .transmit = set_state->model_pub_set.publish_retransmit,
+ };
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_pub_set(&ctx, set_state->model_pub_set.element_addr, set_state->model_pub_set.model_id,
+ set_state->model_pub_set.company_id, &model_pub));
+ }
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_ADD:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_add(&ctx, set_state->model_sub_add.element_addr, set_state->model_sub_add.sub_addr,
+ set_state->model_sub_add.model_id, set_state->model_sub_add.company_id));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_del(&ctx, set_state->model_sub_delete.element_addr, set_state->model_sub_delete.sub_addr,
+ set_state->model_sub_delete.model_id, set_state->model_sub_delete.company_id));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_OVERWRITE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_overwrite(&ctx, set_state->model_sub_overwrite.element_addr, set_state->model_sub_overwrite.sub_addr,
+ set_state->model_sub_overwrite.model_id, set_state->model_sub_overwrite.company_id));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_ADD:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_va_add(&ctx, set_state->model_sub_va_add.element_addr, &set_state->model_sub_va_add.label_uuid[0],
+ set_state->model_sub_va_add.model_id, set_state->model_sub_va_add.company_id));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_OVERWRITE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_va_overwrite(&ctx, set_state->model_sub_va_overwrite.element_addr, &set_state->model_sub_va_overwrite.label_uuid[0],
+ set_state->model_sub_va_overwrite.model_id, set_state->model_sub_va_overwrite.company_id));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_DELETE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_va_del(&ctx, set_state->model_sub_va_delete.element_addr, &set_state->model_sub_va_delete.label_uuid[0],
+ set_state->model_sub_va_delete.model_id, set_state->model_sub_va_delete.company_id));
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_SET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_hb_sub_set(&ctx, (struct bt_mesh_cfg_hb_sub *)&set_state->heartbeat_sub_set));
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_SET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_hb_pub_set(&ctx, (const struct bt_mesh_cfg_hb_pub *)&set_state->heartbeat_pub_set));
+ case ESP_BLE_MESH_MODEL_OP_NODE_RESET:
+ return (cfg_client_cb->error_code = bt_mesh_cfg_node_reset(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_PUB_VIRTUAL_ADDR_SET: {
+ struct bt_mesh_cfg_mod_pub model_pub = {
+ .app_idx = set_state->model_pub_va_set.publish_app_idx,
+ .cred_flag = set_state->model_pub_va_set.cred_flag,
+ .ttl = set_state->model_pub_va_set.publish_ttl,
+ .period = set_state->model_pub_va_set.publish_period,
+ .transmit = set_state->model_pub_va_set.publish_retransmit,
+ };
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_pub_va_set(&ctx, set_state->model_pub_va_set.element_addr, set_state->model_pub_va_set.model_id,
+ set_state->model_pub_va_set.company_id, set_state->model_pub_va_set.label_uuid, &model_pub));
+ }
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE_ALL:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_sub_del_all(&ctx, set_state->model_sub_delete_all.element_addr,
+ set_state->model_sub_delete_all.model_id, set_state->model_sub_delete_all.company_id));
+ case ESP_BLE_MESH_MODEL_OP_NET_KEY_UPDATE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_net_key_update(&ctx, set_state->net_key_update.net_idx, set_state->net_key_update.net_key));
+ case ESP_BLE_MESH_MODEL_OP_NET_KEY_DELETE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_net_key_delete(&ctx, set_state->net_key_delete.net_idx));
+ case ESP_BLE_MESH_MODEL_OP_APP_KEY_UPDATE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_app_key_update(&ctx, set_state->app_key_update.net_idx, set_state->app_key_update.app_idx,
+ set_state->app_key_update.app_key));
+ case ESP_BLE_MESH_MODEL_OP_APP_KEY_DELETE:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_app_key_delete(&ctx, set_state->app_key_delete.net_idx, set_state->app_key_delete.app_idx));
+ case ESP_BLE_MESH_MODEL_OP_NODE_IDENTITY_SET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_node_identity_set(&ctx, set_state->node_identity_set.net_idx, set_state->node_identity_set.identity));
+ case ESP_BLE_MESH_MODEL_OP_MODEL_APP_UNBIND:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_mod_app_unbind(&ctx, set_state->model_app_unbind.element_addr, set_state->model_app_unbind.model_app_idx,
+ set_state->model_app_unbind.model_id, set_state->model_app_unbind.company_id));
+ case ESP_BLE_MESH_MODEL_OP_KEY_REFRESH_PHASE_SET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_kr_phase_set(&ctx, set_state->kr_phase_set.net_idx, set_state->kr_phase_set.transition));
+ case ESP_BLE_MESH_MODEL_OP_NETWORK_TRANSMIT_SET:
+ return (cfg_client_cb->error_code =
+ bt_mesh_cfg_net_transmit_set(&ctx, set_state->net_transmit_set.net_transmit));
+ default:
+ BT_WARN("%s, invalid opcode 0x%x", __func__, params->opcode);
+ return (cfg_client_cb->error_code = -EINVAL);
+ }
+
+ return 0;
+}
+
+static void btc_mesh_cfg_server_callback(esp_ble_mesh_cfg_server_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_CFG_SERVER;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params, sizeof(esp_ble_mesh_cfg_server_cb_param_t), NULL);
+}
+
+void bt_mesh_callback_cfg_server_event_to_btc(u8_t evt_type, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len)
+{
+ esp_ble_mesh_cfg_server_cb_param_t cb_params = {0};
+ size_t length;
+ uint8_t act;
+
+ if (!model || !ctx) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ switch (evt_type) {
+ case 0x00:
+ act = ESP_BLE_MESH_CFG_SERVER_RECV_MSG_EVT;
+ break;
+ default:
+ LOG_ERROR("%s: unknown config sever event type", __func__);
+ return;
+ }
+
+ cb_params.model = (esp_ble_mesh_model_t *)model;
+ cb_params.ctx.net_idx = ctx->net_idx;
+ cb_params.ctx.app_idx = ctx->app_idx;
+ cb_params.ctx.addr = ctx->addr;
+ cb_params.ctx.recv_ttl = ctx->recv_ttl;
+ cb_params.ctx.recv_op = ctx->recv_op;
+ cb_params.ctx.recv_dst = ctx->recv_dst;
+
+ if (val && len) {
+ length = (len <= sizeof(cb_params.status_cb)) ? len : sizeof(cb_params.status_cb);
+ memcpy(&cb_params.status_cb, val, length);
+ }
+
+ btc_mesh_cfg_server_callback(&cb_params, act);
+}
+
+void btc_mesh_cfg_server_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_cfg_server_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_cfg_server_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_CFG_SERVER_EVT_MAX) {
+ btc_ble_mesh_cfg_server_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+}
diff --git a/components/bt/ble_mesh/btc/btc_ble_mesh_generic_client.c b/components/bt/ble_mesh/btc/btc_ble_mesh_generic_client.c
new file mode 100644
index 0000000000..07b2d44db1
--- /dev/null
+++ b/components/bt/ble_mesh/btc/btc_ble_mesh_generic_client.c
@@ -0,0 +1,549 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "cfg_cli.h"
+#include "btc_ble_mesh_generic_client.h"
+#include "esp_ble_mesh_generic_model_api.h"
+
+static inline void btc_ble_mesh_cb_to_app(esp_ble_mesh_generic_client_cb_event_t event,
+ esp_ble_mesh_generic_client_cb_param_t *param)
+{
+ esp_ble_mesh_generic_client_cb_t btc_mesh_cb = (esp_ble_mesh_generic_client_cb_t)btc_profile_cb_get(BTC_PID_GENERIC_CLIENT);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+void btc_ble_mesh_generic_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ btc_ble_mesh_generic_client_args_t *dst = (btc_ble_mesh_generic_client_args_t *)p_dest;
+ btc_ble_mesh_generic_client_args_t *src = (btc_ble_mesh_generic_client_args_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !dst || !src) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_GENERIC_CLIENT_GET_STATE: {
+ dst->generic_client_get_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->generic_client_get_state.get_state = (esp_ble_mesh_generic_client_get_state_t *)osi_malloc(sizeof(esp_ble_mesh_generic_client_get_state_t));
+ if (dst->generic_client_get_state.params && dst->generic_client_get_state.get_state) {
+ memcpy(dst->generic_client_get_state.params, src->generic_client_get_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->generic_client_get_state.get_state, src->generic_client_get_state.get_state,
+ sizeof(esp_ble_mesh_generic_client_get_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_GENERIC_CLIENT_SET_STATE: {
+ dst->generic_client_set_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->generic_client_set_state.set_state = (esp_ble_mesh_generic_client_set_state_t *)osi_malloc(sizeof(esp_ble_mesh_generic_client_set_state_t));
+ if (dst->generic_client_set_state.params && dst->generic_client_set_state.set_state) {
+ memcpy(dst->generic_client_set_state.params, src->generic_client_set_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->generic_client_set_state.set_state, src->generic_client_set_state.set_state,
+ sizeof(esp_ble_mesh_generic_client_set_state_t));
+
+ opcode = src->generic_client_set_state.params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET:
+ if (src->generic_client_set_state.set_state->user_property_set.property_value) {
+ length = src->generic_client_set_state.set_state->user_property_set.property_value->len;
+ dst->generic_client_set_state.set_state->user_property_set.property_value = bt_mesh_alloc_buf(length);
+ if (!dst->generic_client_set_state.set_state->user_property_set.property_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->generic_client_set_state.set_state->user_property_set.property_value, 0);
+ net_buf_simple_add_mem(dst->generic_client_set_state.set_state->user_property_set.property_value,
+ src->generic_client_set_state.set_state->user_property_set.property_value->data,
+ src->generic_client_set_state.set_state->user_property_set.property_value->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET:
+ if (src->generic_client_set_state.set_state->admin_property_set.property_value) {
+ length = src->generic_client_set_state.set_state->admin_property_set.property_value->len;
+ dst->generic_client_set_state.set_state->admin_property_set.property_value = bt_mesh_alloc_buf(length);
+ if (!dst->generic_client_set_state.set_state->admin_property_set.property_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->generic_client_set_state.set_state->admin_property_set.property_value, 0);
+ net_buf_simple_add_mem(dst->generic_client_set_state.set_state->admin_property_set.property_value,
+ src->generic_client_set_state.set_state->admin_property_set.property_value->data,
+ src->generic_client_set_state.set_state->admin_property_set.property_value->len);
+ }
+ break;
+ default:
+ break;
+ }
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ default:
+ LOG_DEBUG("%s Unhandled deep copy %d", __func__, msg->act);
+ break;
+ }
+}
+
+static void btc_ble_mesh_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ esp_ble_mesh_generic_client_cb_param_t *p_dest_data = (esp_ble_mesh_generic_client_cb_param_t *)p_dest;
+ esp_ble_mesh_generic_client_cb_param_t *p_src_data = (esp_ble_mesh_generic_client_cb_param_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !p_src_data || !p_dest_data) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_GENERIC_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_GENERIC_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_GENERIC_CLIENT_PUBLISH_EVT:
+ if (p_src_data->params) {
+ opcode = p_src_data->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS:
+ if (p_src_data->status_cb.user_properties_status.property_ids) {
+ length = p_src_data->status_cb.user_properties_status.property_ids->len;
+ p_dest_data->status_cb.user_properties_status.property_ids = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.user_properties_status.property_ids) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.user_properties_status.property_ids, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.user_properties_status.property_ids,
+ p_src_data->status_cb.user_properties_status.property_ids->data,
+ p_src_data->status_cb.user_properties_status.property_ids->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS:
+ if (p_src_data->status_cb.user_property_status.property_value) {
+ length = p_src_data->status_cb.user_property_status.property_value->len;
+ p_dest_data->status_cb.user_property_status.property_value = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.user_property_status.property_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.user_property_status.property_value, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.user_property_status.property_value,
+ p_src_data->status_cb.user_property_status.property_value->data,
+ p_src_data->status_cb.user_property_status.property_value->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS:
+ if (p_src_data->status_cb.admin_properties_status.property_ids) {
+ length = p_src_data->status_cb.admin_properties_status.property_ids->len;
+ p_dest_data->status_cb.admin_properties_status.property_ids = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.admin_properties_status.property_ids) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.admin_properties_status.property_ids, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.admin_properties_status.property_ids,
+ p_src_data->status_cb.admin_properties_status.property_ids->data,
+ p_src_data->status_cb.admin_properties_status.property_ids->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS:
+ if (p_src_data->status_cb.admin_property_status.property_value) {
+ length = p_src_data->status_cb.admin_property_status.property_value->len;
+ p_dest_data->status_cb.admin_property_status.property_value = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.admin_property_status.property_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.admin_property_status.property_value, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.admin_property_status.property_value,
+ p_src_data->status_cb.admin_property_status.property_value->data,
+ p_src_data->status_cb.admin_property_status.property_value->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTIES_STATUS:
+ if (p_src_data->status_cb.manufacturer_properties_status.property_ids) {
+ length = p_src_data->status_cb.manufacturer_properties_status.property_ids->len;
+ p_dest_data->status_cb.manufacturer_properties_status.property_ids = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.manufacturer_properties_status.property_ids) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.manufacturer_properties_status.property_ids, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.manufacturer_properties_status.property_ids,
+ p_src_data->status_cb.manufacturer_properties_status.property_ids->data,
+ p_src_data->status_cb.manufacturer_properties_status.property_ids->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_STATUS:
+ if (p_src_data->status_cb.manufacturer_property_status.property_value) {
+ length = p_src_data->status_cb.manufacturer_property_status.property_value->len;
+ p_dest_data->status_cb.manufacturer_property_status.property_value = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.manufacturer_property_status.property_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.manufacturer_property_status.property_value, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.manufacturer_property_status.property_value,
+ p_src_data->status_cb.manufacturer_property_status.property_value->data,
+ p_src_data->status_cb.manufacturer_property_status.property_value->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS:
+ if (p_src_data->status_cb.client_properties_status.property_ids) {
+ length = p_src_data->status_cb.client_properties_status.property_ids->len;
+ p_dest_data->status_cb.client_properties_status.property_ids = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.client_properties_status.property_ids) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.client_properties_status.property_ids, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.client_properties_status.property_ids,
+ p_src_data->status_cb.client_properties_status.property_ids->data,
+ p_src_data->status_cb.client_properties_status.property_ids->len);
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_GENERIC_CLIENT_TIMEOUT_EVT:
+ if (p_src_data->params) {
+ p_dest_data->params = osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ if (p_dest_data->params) {
+ memcpy(p_dest_data->params, p_src_data->params, sizeof(esp_ble_mesh_client_common_param_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_free_req_data(btc_msg_t *msg)
+{
+ esp_ble_mesh_generic_client_cb_param_t *arg = NULL;
+ u32_t opcode;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (esp_ble_mesh_generic_client_cb_param_t *)(msg->arg);
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_GENERIC_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_GENERIC_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_GENERIC_CLIENT_PUBLISH_EVT:
+ if (arg->params) {
+ opcode = arg->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS:
+ bt_mesh_free_buf(arg->status_cb.user_properties_status.property_ids);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS:
+ bt_mesh_free_buf(arg->status_cb.user_property_status.property_value);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS:
+ bt_mesh_free_buf(arg->status_cb.admin_properties_status.property_ids);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS:
+ bt_mesh_free_buf(arg->status_cb.admin_property_status.property_value);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTIES_STATUS:
+ bt_mesh_free_buf(arg->status_cb.manufacturer_properties_status.property_ids);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_MANUFACTURER_PROPERTY_STATUS:
+ bt_mesh_free_buf(arg->status_cb.manufacturer_property_status.property_value);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS:
+ bt_mesh_free_buf(arg->status_cb.client_properties_status.property_ids);
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_GENERIC_CLIENT_TIMEOUT_EVT:
+ if (arg->params) {
+ osi_free(arg->params);
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+void btc_ble_mesh_generic_client_arg_deep_free(btc_msg_t *msg)
+{
+ btc_ble_mesh_generic_client_args_t *arg = NULL;
+ u32_t opcode = 0;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_generic_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_GENERIC_CLIENT_GET_STATE:
+ if (arg->generic_client_get_state.params) {
+ osi_free(arg->generic_client_get_state.params);
+ }
+ if (arg->generic_client_get_state.get_state) {
+ osi_free(arg->generic_client_get_state.get_state);
+ }
+ break;
+ case BTC_BLE_MESH_ACT_GENERIC_CLIENT_SET_STATE:
+ if (arg->generic_client_set_state.params) {
+ opcode = arg->generic_client_set_state.params->opcode;
+ osi_free(arg->generic_client_set_state.params);
+ }
+ if (arg->generic_client_set_state.set_state) {
+ if (opcode) {
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_USER_PROPERTY_SET:
+ bt_mesh_free_buf(arg->generic_client_set_state.set_state->user_property_set.property_value);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET:
+ bt_mesh_free_buf(arg->generic_client_set_state.set_state->admin_property_set.property_value);
+ break;
+ default:
+ break;
+ }
+ }
+ osi_free(arg->generic_client_set_state.set_state);
+ }
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+static void btc_mesh_generic_client_callback(esp_ble_mesh_generic_client_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_GENERIC_CLIENT;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params,
+ sizeof(esp_ble_mesh_generic_client_cb_param_t), btc_ble_mesh_copy_req_data);
+}
+
+void bt_mesh_callback_generic_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len)
+{
+ esp_ble_mesh_generic_client_cb_param_t cb_params = {0};
+ esp_ble_mesh_client_common_param_t params = {0};
+ size_t length;
+ uint8_t act;
+
+ if (!model || !ctx) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ switch (evt_type) {
+ case 0x00:
+ act = ESP_BLE_MESH_GENERIC_CLIENT_GET_STATE_EVT;
+ break;
+ case 0x01:
+ act = ESP_BLE_MESH_GENERIC_CLIENT_SET_STATE_EVT;
+ break;
+ case 0x02:
+ act = ESP_BLE_MESH_GENERIC_CLIENT_PUBLISH_EVT;
+ break;
+ case 0x03:
+ act = ESP_BLE_MESH_GENERIC_CLIENT_TIMEOUT_EVT;
+ break;
+ default:
+ LOG_ERROR("%s: unknown generic status event type", __func__);
+ return;
+ }
+
+ params.opcode = opcode;
+ params.model = (esp_ble_mesh_model_t *)model;
+ params.ctx.net_idx = ctx->net_idx;
+ params.ctx.app_idx = ctx->app_idx;
+ params.ctx.addr = ctx->addr;
+ params.ctx.recv_ttl = ctx->recv_ttl;
+ params.ctx.recv_op = ctx->recv_op;
+ params.ctx.recv_dst = ctx->recv_dst;
+
+ cb_params.error_code = 0;
+ cb_params.params = ¶ms;
+
+ if (val && len) {
+ length = (len <= sizeof(cb_params.status_cb)) ? len : sizeof(cb_params.status_cb);
+ memcpy(&cb_params.status_cb, val, length);
+ }
+
+ btc_mesh_generic_client_callback(&cb_params, act);
+}
+
+void btc_mesh_generic_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf)
+{
+ if (!model || !ctx || !buf) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ bt_mesh_callback_generic_status_to_btc(opcode, 0x02, model, ctx, buf->data, buf->len);
+}
+
+void btc_mesh_generic_client_call_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_generic_client_cb_param_t generic_client_cb = {0};
+ esp_ble_mesh_client_common_param_t *params = NULL;
+ btc_ble_mesh_generic_client_args_t *arg = NULL;
+ struct bt_mesh_common_param common = {0};
+ bt_mesh_role_param_t role_param = {0};
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_generic_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_GENERIC_CLIENT_GET_STATE: {
+ params = arg->generic_client_get_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ generic_client_cb.params = arg->generic_client_get_state.params;
+ generic_client_cb.error_code =
+ bt_mesh_generic_client_get_state(&common,
+ (void *)arg->generic_client_get_state.get_state,
+ (void *)&generic_client_cb.status_cb);
+ if (generic_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_generic_client_callback(&generic_client_cb,
+ ESP_BLE_MESH_GENERIC_CLIENT_GET_STATE_EVT);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_GENERIC_CLIENT_SET_STATE: {
+ params = arg->generic_client_set_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ generic_client_cb.params = arg->generic_client_set_state.params;
+ generic_client_cb.error_code =
+ bt_mesh_generic_client_set_state(&common,
+ (void *)arg->generic_client_set_state.set_state,
+ (void *)&generic_client_cb.status_cb);
+ if (generic_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_generic_client_callback(&generic_client_cb,
+ ESP_BLE_MESH_GENERIC_CLIENT_SET_STATE_EVT);
+ }
+ break;
+ }
+ default:
+ break;
+ }
+
+ btc_ble_mesh_generic_client_arg_deep_free(msg);
+}
+
+void btc_mesh_generic_client_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_generic_client_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_generic_client_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_GENERIC_CLIENT_EVT_MAX) {
+ btc_ble_mesh_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_free_req_data(msg);
+}
diff --git a/components/bt/ble_mesh/btc/btc_ble_mesh_health.c b/components/bt/ble_mesh/btc/btc_ble_mesh_health.c
new file mode 100644
index 0000000000..7f4e75e439
--- /dev/null
+++ b/components/bt/ble_mesh/btc/btc_ble_mesh_health.c
@@ -0,0 +1,593 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_manage.h"
+#include "btc/btc_task.h"
+#include "osi/allocator.h"
+
+#include "health_srv.h"
+#include "health_cli.h"
+#include "common.h"
+#include "btc_ble_mesh_health.h"
+#include "esp_ble_mesh_defs.h"
+
+extern s32_t health_msg_timeout;
+
+static inline void btc_ble_mesh_health_client_cb_to_app(esp_ble_mesh_health_client_cb_event_t event,
+ esp_ble_mesh_health_client_cb_param_t *param)
+{
+ esp_ble_mesh_health_client_cb_t btc_mesh_cb = (esp_ble_mesh_health_client_cb_t)btc_profile_cb_get(BTC_PID_HEALTH_CLIENT);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+static inline void btc_ble_mesh_health_server_cb_to_app(esp_ble_mesh_health_server_cb_event_t event,
+ esp_ble_mesh_health_server_cb_param_t *param)
+{
+ esp_ble_mesh_health_server_cb_t btc_mesh_cb = (esp_ble_mesh_health_server_cb_t)btc_profile_cb_get(BTC_PID_HEALTH_SERVER);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+void btc_ble_mesh_health_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ btc_ble_mesh_health_client_args_t *dst = (btc_ble_mesh_health_client_args_t *)p_dest;
+ btc_ble_mesh_health_client_args_t *src = (btc_ble_mesh_health_client_args_t *)p_src;
+
+ if (!msg || !dst || !src) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_HEALTH_CLIENT_GET_STATE: {
+ dst->health_client_get_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->health_client_get_state.get_state = (esp_ble_mesh_health_client_get_state_t *)osi_malloc(sizeof(esp_ble_mesh_health_client_get_state_t));
+ if (dst->health_client_get_state.params && dst->health_client_get_state.get_state) {
+ memcpy(dst->health_client_get_state.params, src->health_client_get_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->health_client_get_state.get_state, src->health_client_get_state.get_state,
+ sizeof(esp_ble_mesh_health_client_get_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_HEALTH_CLIENT_SET_STATE: {
+ dst->health_client_set_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->health_client_set_state.set_state = (esp_ble_mesh_health_client_set_state_t *)osi_malloc(sizeof(esp_ble_mesh_health_client_set_state_t));
+ if (dst->health_client_set_state.params && dst->health_client_set_state.set_state) {
+ memcpy(dst->health_client_set_state.params, src->health_client_set_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->health_client_set_state.set_state, src->health_client_set_state.set_state,
+ sizeof(esp_ble_mesh_health_client_set_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ default:
+ LOG_DEBUG("%s Unhandled deep copy %d", __func__, msg->act);
+ break;
+ }
+}
+
+static void btc_ble_mesh_health_client_arg_deep_free(btc_msg_t *msg)
+{
+ btc_ble_mesh_health_client_args_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_health_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_HEALTH_CLIENT_GET_STATE:
+ if (arg->health_client_get_state.params) {
+ osi_free(arg->health_client_get_state.params);
+ }
+ if (arg->health_client_get_state.get_state) {
+ osi_free(arg->health_client_get_state.get_state);
+ }
+ break;
+ case BTC_BLE_MESH_ACT_HEALTH_CLIENT_SET_STATE:
+ if (arg->health_client_set_state.params) {
+ osi_free(arg->health_client_set_state.params);
+ }
+ if (arg->health_client_set_state.set_state) {
+ osi_free(arg->health_client_set_state.set_state);
+ }
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+void btc_ble_mesh_health_server_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ if (!msg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_HEALTH_SERVER_FAULT_UPDATE:
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_health_server_arg_deep_free(btc_msg_t *msg)
+{
+ if (!msg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_HEALTH_SERVER_FAULT_UPDATE:
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_health_client_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ esp_ble_mesh_health_client_cb_param_t *p_dest_data = (esp_ble_mesh_health_client_cb_param_t *)p_dest;
+ esp_ble_mesh_health_client_cb_param_t *p_src_data = (esp_ble_mesh_health_client_cb_param_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !p_src_data || !p_dest_data) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_HEALTH_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_HEALTH_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_HEALTH_CLIENT_PUBLISH_EVT:
+ if (p_src_data->params) {
+ opcode = p_src_data->params->opcode;
+ switch (opcode) {
+ case OP_HEALTH_CURRENT_STATUS:
+ if (p_src_data->status_cb.current_status.fault_array) {
+ length = p_src_data->status_cb.current_status.fault_array->len;
+ p_dest_data->status_cb.current_status.fault_array = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.current_status.fault_array) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.current_status.fault_array, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.current_status.fault_array,
+ p_src_data->status_cb.current_status.fault_array->data,
+ p_src_data->status_cb.current_status.fault_array->len);
+ }
+ break;
+ case OP_HEALTH_FAULT_GET:
+ case OP_HEALTH_FAULT_CLEAR:
+ case OP_HEALTH_FAULT_TEST:
+ case OP_HEALTH_FAULT_STATUS:
+ if (p_src_data->status_cb.fault_status.fault_array) {
+ length = p_src_data->status_cb.fault_status.fault_array->len;
+ p_dest_data->status_cb.fault_status.fault_array = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.fault_status.fault_array) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.fault_status.fault_array, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.fault_status.fault_array,
+ p_src_data->status_cb.fault_status.fault_array->data,
+ p_src_data->status_cb.fault_status.fault_array->len);
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_HEALTH_CLIENT_TIMEOUT_EVT:
+ if (p_src_data->params) {
+ p_dest_data->params = osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ if (p_dest_data->params) {
+ memcpy(p_dest_data->params, p_src_data->params, sizeof(esp_ble_mesh_client_common_param_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_health_client_free_req_data(btc_msg_t *msg)
+{
+ esp_ble_mesh_health_client_cb_param_t *arg = NULL;
+ u32_t opcode;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (esp_ble_mesh_health_client_cb_param_t *)(msg->arg);
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_HEALTH_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_HEALTH_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_HEALTH_CLIENT_PUBLISH_EVT:
+ if (arg->params) {
+ opcode = arg->params->opcode;
+ switch (opcode) {
+ case OP_HEALTH_CURRENT_STATUS:
+ bt_mesh_free_buf(arg->status_cb.current_status.fault_array);
+ break;
+ case OP_HEALTH_FAULT_GET:
+ case OP_HEALTH_FAULT_CLEAR:
+ case OP_HEALTH_FAULT_TEST:
+ case OP_HEALTH_FAULT_STATUS:
+ bt_mesh_free_buf(arg->status_cb.fault_status.fault_array);
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_HEALTH_CLIENT_TIMEOUT_EVT:
+ if (arg->params) {
+ osi_free(arg->params);
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_health_server_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ if (!msg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_HEALTH_SERVER_FAULT_UPDATE_COMPLETE_EVT:
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_health_server_free_req_data(btc_msg_t *msg)
+{
+ if (!msg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_HEALTH_SERVER_FAULT_UPDATE_COMPLETE_EVT:
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_mesh_health_client_callback(esp_ble_mesh_health_client_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_HEALTH_CLIENT;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params,
+ sizeof(esp_ble_mesh_health_client_cb_param_t), btc_ble_mesh_health_client_copy_req_data);
+}
+
+static void btc_mesh_health_server_callback(esp_ble_mesh_health_server_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_HEALTH_SERVER;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params,
+ sizeof(esp_ble_mesh_health_server_cb_param_t), btc_ble_mesh_health_server_copy_req_data);
+}
+
+int btc_ble_mesh_health_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_get_state_t *get_state,
+ esp_ble_mesh_health_client_cb_param_t *client_cb)
+{
+ struct bt_mesh_msg_ctx ctx = {0};
+
+ if (!params || !client_cb) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ ctx.net_idx = params->ctx.net_idx;
+ ctx.app_idx = params->ctx.app_idx;
+ ctx.addr = params->ctx.addr;
+ ctx.send_rel = params->ctx.send_rel;
+ ctx.send_ttl = params->ctx.send_ttl;
+
+ health_msg_timeout = params->msg_timeout;
+
+ switch (params->opcode) {
+ case ESP_BLE_MESH_MODEL_OP_ATTENTION_GET:
+ return (client_cb->error_code = bt_mesh_health_attention_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_GET:
+ return (client_cb->error_code = bt_mesh_health_period_get(&ctx));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_GET:
+ return (client_cb->error_code = bt_mesh_health_fault_get(&ctx, get_state->fault_get.company_id));
+ default:
+ BT_WARN("%s, invalid opcode 0x%x", __func__, params->opcode);
+ return (client_cb->error_code = -EINVAL);
+ }
+
+ return 0;
+}
+
+int btc_ble_mesh_health_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_set_state_t *set_state,
+ esp_ble_mesh_health_client_cb_param_t *client_cb)
+{
+ struct bt_mesh_msg_ctx ctx = {0};
+
+ if (!params || !set_state || !client_cb) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ ctx.net_idx = params->ctx.net_idx;
+ ctx.app_idx = params->ctx.app_idx;
+ ctx.addr = params->ctx.addr;
+ ctx.send_rel = params->ctx.send_rel;
+ ctx.send_ttl = params->ctx.send_ttl;
+
+ health_msg_timeout = params->msg_timeout;
+
+ switch (params->opcode) {
+ case ESP_BLE_MESH_MODEL_OP_ATTENTION_SET:
+ return (client_cb->error_code =
+ bt_mesh_health_attention_set(&ctx, set_state->attention_set.attention, true));
+ case ESP_BLE_MESH_MODEL_OP_ATTENTION_SET_UNACK:
+ return (client_cb->error_code =
+ bt_mesh_health_attention_set(&ctx, set_state->attention_set.attention, false));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET:
+ return (client_cb->error_code =
+ bt_mesh_health_period_set(&ctx, set_state->period_set.fast_period_divisor, true));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_PERIOD_SET_UNACK:
+ return (client_cb->error_code =
+ bt_mesh_health_period_set(&ctx, set_state->period_set.fast_period_divisor, false));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST:
+ return (client_cb->error_code =
+ bt_mesh_health_fault_test(&ctx, set_state->fault_test.company_id, set_state->fault_test.test_id, true));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_TEST_UNACK:
+ return (client_cb->error_code =
+ bt_mesh_health_fault_test(&ctx, set_state->fault_test.company_id, set_state->fault_test.test_id, false));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR:
+ return (client_cb->error_code =
+ bt_mesh_health_fault_clear(&ctx, set_state->fault_clear.company_id, true));
+ case ESP_BLE_MESH_MODEL_OP_HEALTH_FAULT_CLEAR_UNACK:
+ return (client_cb->error_code =
+ bt_mesh_health_fault_clear(&ctx, set_state->fault_clear.company_id, false));
+ default:
+ BT_WARN("%s, invalid opcode 0x%x", __func__, params->opcode);
+ return (client_cb->error_code = -EINVAL);
+ }
+
+ return 0;
+}
+
+void bt_mesh_callback_health_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, u16_t len)
+{
+ esp_ble_mesh_health_client_cb_param_t cb_params = {0};
+ esp_ble_mesh_client_common_param_t params = {0};
+ size_t length;
+ uint8_t act;
+
+ if (!model || !ctx) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ switch (evt_type) {
+ case 0x00:
+ act = ESP_BLE_MESH_HEALTH_CLIENT_GET_STATE_EVT;
+ break;
+ case 0x01:
+ act = ESP_BLE_MESH_HEALTH_CLIENT_SET_STATE_EVT;
+ break;
+ case 0x02:
+ act = ESP_BLE_MESH_HEALTH_CLIENT_PUBLISH_EVT;
+ break;
+ case 0x03:
+ act = ESP_BLE_MESH_HEALTH_CLIENT_TIMEOUT_EVT;
+ break;
+ default:
+ LOG_ERROR("%s: unknown health status event type", __func__);
+ return;
+ }
+
+ params.opcode = opcode;
+ params.model = (esp_ble_mesh_model_t *)model;
+ params.ctx.net_idx = ctx->net_idx;
+ params.ctx.app_idx = ctx->app_idx;
+ params.ctx.addr = ctx->addr;
+ params.ctx.recv_ttl = ctx->recv_ttl;
+ params.ctx.recv_op = ctx->recv_op;
+ params.ctx.recv_dst = ctx->recv_dst;
+
+ cb_params.error_code = 0;
+ cb_params.params = ¶ms;
+
+ if (val && len) {
+ length = (len <= sizeof(cb_params.status_cb)) ? len : sizeof(cb_params.status_cb);
+ memcpy(&cb_params.status_cb, val, length);
+ }
+
+ btc_mesh_health_client_callback(&cb_params, act);
+}
+
+void btc_mesh_health_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf)
+{
+ if (!model || !ctx || !buf) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ bt_mesh_callback_health_status_to_btc(opcode, 0x02, model, ctx, buf->data, buf->len);
+}
+
+void btc_mesh_health_client_call_handler(btc_msg_t *msg)
+{
+ btc_ble_mesh_health_client_args_t *arg = NULL;
+ esp_ble_mesh_health_client_cb_param_t health_client_cb = {0};
+ bt_mesh_role_param_t role_param = {0};
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_health_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_HEALTH_CLIENT_GET_STATE: {
+ health_client_cb.params = arg->health_client_get_state.params;
+ role_param.model = (struct bt_mesh_model *)health_client_cb.params->model;
+ role_param.role = health_client_cb.params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ btc_ble_mesh_health_client_get_state(arg->health_client_get_state.params,
+ arg->health_client_get_state.get_state,
+ &health_client_cb);
+ if (health_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_health_client_callback(&health_client_cb, ESP_BLE_MESH_HEALTH_CLIENT_GET_STATE_EVT);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_HEALTH_CLIENT_SET_STATE: {
+ health_client_cb.params = arg->health_client_set_state.params;
+ role_param.model = (struct bt_mesh_model *)health_client_cb.params->model;
+ role_param.role = health_client_cb.params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ btc_ble_mesh_health_client_set_state(arg->health_client_set_state.params,
+ arg->health_client_set_state.set_state,
+ &health_client_cb);
+ if (health_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_health_client_callback(&health_client_cb, ESP_BLE_MESH_HEALTH_CLIENT_SET_STATE_EVT);
+ }
+ break;
+ }
+ default:
+ break;
+ }
+
+ btc_ble_mesh_health_client_arg_deep_free(msg);
+ return;
+}
+
+void btc_mesh_health_client_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_health_client_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_health_client_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_HEALTH_CLIENT_EVT_MAX) {
+ btc_ble_mesh_health_client_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_health_client_free_req_data(msg);
+}
+
+void btc_mesh_health_server_call_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_health_server_cb_param_t health_server_cb = {0};
+ btc_ble_mesh_health_server_args_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_health_server_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_HEALTH_SERVER_FAULT_UPDATE: {
+ health_server_cb.error_code = bt_mesh_fault_update((struct bt_mesh_elem *)arg->fault_update.element);
+ btc_mesh_health_server_callback(&health_server_cb, ESP_BLE_MESH_HEALTH_SERVER_FAULT_UPDATE_COMPLETE_EVT);
+ }
+ default:
+ break;
+ }
+
+ btc_ble_mesh_health_server_arg_deep_free(msg);
+}
+
+void btc_mesh_health_server_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_health_server_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_health_server_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_HEALTH_SERVER_EVT_MAX) {
+ btc_ble_mesh_health_server_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_health_server_free_req_data(msg);
+}
diff --git a/components/bt/ble_mesh/btc/btc_ble_mesh_light_client.c b/components/bt/ble_mesh/btc/btc_ble_mesh_light_client.c
new file mode 100644
index 0000000000..b5c88d729e
--- /dev/null
+++ b/components/bt/ble_mesh/btc/btc_ble_mesh_light_client.c
@@ -0,0 +1,382 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "light_client.h"
+#include "btc_ble_mesh_light_client.h"
+#include "esp_ble_mesh_lighting_model_api.h"
+
+static inline void btc_ble_mesh_cb_to_app(esp_ble_mesh_light_client_cb_event_t event,
+ esp_ble_mesh_light_client_cb_param_t *param)
+{
+ esp_ble_mesh_light_client_cb_t btc_mesh_cb = (esp_ble_mesh_light_client_cb_t)btc_profile_cb_get(BTC_PID_LIGHT_CLIENT);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+void btc_ble_mesh_light_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ btc_ble_mesh_light_client_args_t *dst = (btc_ble_mesh_light_client_args_t *)p_dest;
+ btc_ble_mesh_light_client_args_t *src = (btc_ble_mesh_light_client_args_t *)p_src;
+
+ if (!msg || !dst || !src) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_LIGHT_CLIENT_GET_STATE: {
+ dst->light_client_get_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->light_client_get_state.get_state = (esp_ble_mesh_light_client_get_state_t *)osi_malloc(sizeof(esp_ble_mesh_light_client_get_state_t));
+ if (dst->light_client_get_state.params && dst->light_client_get_state.get_state) {
+ memcpy(dst->light_client_get_state.params, src->light_client_get_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->light_client_get_state.get_state, src->light_client_get_state.get_state,
+ sizeof(esp_ble_mesh_light_client_get_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_LIGHT_CLIENT_SET_STATE: {
+ dst->light_client_set_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->light_client_set_state.set_state = (esp_ble_mesh_light_client_set_state_t *)osi_malloc(sizeof(esp_ble_mesh_light_client_set_state_t));
+ if (dst->light_client_set_state.params && dst->light_client_set_state.set_state) {
+ memcpy(dst->light_client_set_state.params, src->light_client_set_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->light_client_set_state.set_state, src->light_client_set_state.set_state,
+ sizeof(esp_ble_mesh_light_client_set_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ default:
+ LOG_DEBUG("%s Unhandled deep copy %d", __func__, msg->act);
+ break;
+ }
+}
+
+static void btc_ble_mesh_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ esp_ble_mesh_light_client_cb_param_t *p_dest_data = (esp_ble_mesh_light_client_cb_param_t *)p_dest;
+ esp_ble_mesh_light_client_cb_param_t *p_src_data = (esp_ble_mesh_light_client_cb_param_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !p_src_data || !p_dest_data) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_LIGHT_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_LIGHT_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_LIGHT_CLIENT_PUBLISH_EVT:
+ if (p_src_data->params) {
+ opcode = p_src_data->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS:
+ if (p_src_data->status_cb.lc_property_status.property_value) {
+ length = p_src_data->status_cb.lc_property_status.property_value->len;
+ p_dest_data->status_cb.lc_property_status.property_value = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.lc_property_status.property_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.lc_property_status.property_value, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.lc_property_status.property_value,
+ p_src_data->status_cb.lc_property_status.property_value->data,
+ p_src_data->status_cb.lc_property_status.property_value->len);
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_LIGHT_CLIENT_TIMEOUT_EVT:
+ if (p_src_data->params) {
+ p_dest_data->params = osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ if (p_dest_data->params) {
+ memcpy(p_dest_data->params, p_src_data->params, sizeof(esp_ble_mesh_client_common_param_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_free_req_data(btc_msg_t *msg)
+{
+ esp_ble_mesh_light_client_cb_param_t *arg = NULL;
+ u32_t opcode;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (esp_ble_mesh_light_client_cb_param_t *)(msg->arg);
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_LIGHT_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_LIGHT_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_LIGHT_CLIENT_PUBLISH_EVT:
+ if (arg->params) {
+ opcode = arg->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET:
+ case ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
+ case ESP_BLE_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS:
+ bt_mesh_free_buf(arg->status_cb.lc_property_status.property_value);
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_LIGHT_CLIENT_TIMEOUT_EVT:
+ if (arg->params) {
+ osi_free(arg->params);
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+void btc_ble_mesh_light_client_arg_deep_free(btc_msg_t *msg)
+{
+ btc_ble_mesh_light_client_args_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_light_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_LIGHT_CLIENT_GET_STATE:
+ if (arg->light_client_get_state.params) {
+ osi_free(arg->light_client_get_state.params);
+ }
+ if (arg->light_client_get_state.get_state) {
+ osi_free(arg->light_client_get_state.get_state);
+ }
+ break;
+ case BTC_BLE_MESH_ACT_LIGHT_CLIENT_SET_STATE:
+ if (arg->light_client_set_state.params) {
+ osi_free(arg->light_client_set_state.params);
+ }
+ if (arg->light_client_set_state.set_state) {
+ osi_free(arg->light_client_set_state.set_state);
+ }
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+static void btc_mesh_light_client_callback(esp_ble_mesh_light_client_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_LIGHT_CLIENT;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params,
+ sizeof(esp_ble_mesh_light_client_cb_param_t), btc_ble_mesh_copy_req_data);
+}
+
+void bt_mesh_callback_light_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len)
+{
+ esp_ble_mesh_light_client_cb_param_t cb_params = {0};
+ esp_ble_mesh_client_common_param_t params = {0};
+ size_t length;
+ uint8_t act;
+
+ if (!model || !ctx) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ switch (evt_type) {
+ case 0x00:
+ act = ESP_BLE_MESH_LIGHT_CLIENT_GET_STATE_EVT;
+ break;
+ case 0x01:
+ act = ESP_BLE_MESH_LIGHT_CLIENT_SET_STATE_EVT;
+ break;
+ case 0x02:
+ act = ESP_BLE_MESH_LIGHT_CLIENT_PUBLISH_EVT;
+ break;
+ case 0x03:
+ act = ESP_BLE_MESH_LIGHT_CLIENT_TIMEOUT_EVT;
+ break;
+ default:
+ LOG_ERROR("%s: unknown light status event type", __func__);
+ return;
+ }
+
+ params.opcode = opcode;
+ params.model = (esp_ble_mesh_model_t *)model;
+ params.ctx.net_idx = ctx->net_idx;
+ params.ctx.app_idx = ctx->app_idx;
+ params.ctx.addr = ctx->addr;
+ params.ctx.recv_ttl = ctx->recv_ttl;
+ params.ctx.recv_op = ctx->recv_op;
+ params.ctx.recv_dst = ctx->recv_dst;
+
+ cb_params.error_code = 0;
+ cb_params.params = ¶ms;
+
+ if (val && len) {
+ length = (len <= sizeof(cb_params.status_cb)) ? len : sizeof(cb_params.status_cb);
+ memcpy(&cb_params.status_cb, val, length);
+ }
+
+ btc_mesh_light_client_callback(&cb_params, act);
+}
+
+void btc_mesh_light_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf)
+{
+ if (!model || !ctx || !buf) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ bt_mesh_callback_light_status_to_btc(opcode, 0x02, model, ctx, buf->data, buf->len);
+}
+
+void btc_mesh_light_client_call_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_light_client_cb_param_t light_client_cb = {0};
+ esp_ble_mesh_client_common_param_t *params = NULL;
+ btc_ble_mesh_light_client_args_t *arg = NULL;
+ struct bt_mesh_common_param common = {0};
+ bt_mesh_role_param_t role_param = {0};
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_light_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_LIGHT_CLIENT_GET_STATE: {
+ params = arg->light_client_get_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ light_client_cb.params = arg->light_client_get_state.params;
+ light_client_cb.error_code =
+ bt_mesh_light_client_get_state(&common,
+ (void *)arg->light_client_get_state.get_state,
+ (void *)&light_client_cb.status_cb);
+ if (light_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_light_client_callback(&light_client_cb,
+ ESP_BLE_MESH_LIGHT_CLIENT_GET_STATE_EVT);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_LIGHT_CLIENT_SET_STATE: {
+ params = arg->light_client_set_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ light_client_cb.params = arg->light_client_set_state.params;
+ light_client_cb.error_code =
+ bt_mesh_light_client_set_state(&common,
+ (void *)arg->light_client_set_state.set_state,
+ (void *)&light_client_cb.status_cb);
+ if (light_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_light_client_callback(&light_client_cb,
+ ESP_BLE_MESH_LIGHT_CLIENT_SET_STATE_EVT);
+ }
+ break;
+ }
+ default:
+ break;
+ }
+
+ btc_ble_mesh_light_client_arg_deep_free(msg);
+}
+
+void btc_mesh_light_client_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_light_client_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_light_client_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_LIGHT_CLIENT_EVT_MAX) {
+ btc_ble_mesh_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_free_req_data(msg);
+}
+
diff --git a/components/bt/ble_mesh/btc/btc_ble_mesh_prov.c b/components/bt/ble_mesh/btc/btc_ble_mesh_prov.c
new file mode 100644
index 0000000000..b5db26ad08
--- /dev/null
+++ b/components/bt/ble_mesh/btc/btc_ble_mesh_prov.c
@@ -0,0 +1,1504 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "freertos/queue.h"
+#include "freertos/semphr.h"
+
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+#include "btc_ble_mesh_prov.h"
+
+#include "sdkconfig.h"
+
+#include "mesh.h"
+#include "mesh_util.h"
+#include "mesh_main.h"
+#include "mesh_access.h"
+#include "access.h"
+#include "transport.h"
+#include "proxy.h"
+#include "prov.h"
+#include "include/proxy.h"
+
+#include "cfg_cli.h"
+#include "health_cli.h"
+#include "generic_client.h"
+#include "light_client.h"
+#include "sensor_client.h"
+#include "time_scene_client.h"
+#include "provisioner_prov.h"
+#include "provisioner_main.h"
+#include "bt_mesh_client_common.h"
+
+#include "btc_ble_mesh_config_client.h"
+#include "btc_ble_mesh_health.h"
+#include "btc_ble_mesh_generic_client.h"
+#include "btc_ble_mesh_time_scene_client.h"
+#include "btc_ble_mesh_sensor_client.h"
+#include "btc_ble_mesh_light_client.h"
+
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_common_api.h"
+#include "esp_ble_mesh_provisioning_api.h"
+#include "esp_ble_mesh_networking_api.h"
+
+#if CONFIG_BT_MESH
+
+#define BLE_MESH_MAX_DATA_SIZE 379
+
+void btc_ble_mesh_prov_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ btc_ble_mesh_model_args_t *dst = (btc_ble_mesh_model_args_t *)p_dest;
+ btc_ble_mesh_model_args_t *src = (btc_ble_mesh_model_args_t *)p_src;
+
+ if (!msg || !dst || !src) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_SERVER_MODEL_SEND:
+ case BTC_BLE_MESH_ACT_CLIENT_MODEL_SEND: {
+ LOG_DEBUG("%s, BTC_BLE_MESH_ACT_MODEL_SEND, src->model_send.length = %d", __func__, src->model_send.length);
+ dst->model_send.data = src->model_send.length ? (uint8_t *)osi_malloc(src->model_send.length) : NULL;
+ dst->model_send.ctx = osi_malloc(sizeof(esp_ble_mesh_msg_ctx_t));
+ if (src->model_send.length) {
+ if (dst->model_send.data) {
+ memcpy(dst->model_send.data, src->model_send.data, src->model_send.length);
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ if (dst->model_send.ctx) {
+ memcpy(dst->model_send.ctx, src->model_send.ctx, sizeof(esp_ble_mesh_msg_ctx_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ LOG_DEBUG("dst->model_send.data = %s", bt_hex(dst->model_send.data, src->model_send.length));
+ break;
+ }
+ default:
+ LOG_DEBUG("%s Unhandled deep copy %d", __func__, msg->act);
+ break;
+ }
+}
+
+void btc_ble_mesh_prov_arg_deep_free(btc_msg_t *msg)
+{
+ btc_ble_mesh_model_args_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_model_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_SERVER_MODEL_SEND:
+ case BTC_BLE_MESH_ACT_CLIENT_MODEL_SEND:
+ if (arg->model_send.data) {
+ osi_free(arg->model_send.data);
+ }
+ if (arg->model_send.ctx) {
+ osi_free(arg->model_send.ctx);
+ }
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+static void btc_ble_mesh_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ esp_ble_mesh_model_cb_param_t *p_dest_data = (esp_ble_mesh_model_cb_param_t *)p_dest;
+ esp_ble_mesh_model_cb_param_t *p_src_data = (esp_ble_mesh_model_cb_param_t *)p_src;
+
+ if (!msg || !p_src_data || !p_dest_data) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_MODEL_OPERATION_EVT: {
+ if (p_src_data->model_operation.ctx && p_src_data->model_operation.msg) {
+ p_dest_data->model_operation.ctx = osi_malloc(sizeof(esp_ble_mesh_msg_ctx_t));
+ p_dest_data->model_operation.msg = p_src_data->model_operation.length ? (uint8_t *)osi_malloc(p_src_data->model_operation.length) : NULL;
+ if (p_dest_data->model_operation.ctx) {
+ memcpy(p_dest_data->model_operation.ctx, p_src_data->model_operation.ctx, sizeof(esp_ble_mesh_msg_ctx_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ if (p_src_data->model_operation.length) {
+ if (p_dest_data->model_operation.msg) {
+ memcpy(p_dest_data->model_operation.msg, p_src_data->model_operation.msg, p_src_data->model_operation.length);
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ }
+ break;
+ }
+ case ESP_BLE_MESH_CLIENT_MODEL_RECV_PUBLISH_MSG_EVT: {
+ if (p_src_data->client_recv_publish_msg.ctx && p_src_data->client_recv_publish_msg.msg) {
+ p_dest_data->client_recv_publish_msg.ctx = osi_malloc(sizeof(esp_ble_mesh_msg_ctx_t));
+ p_dest_data->client_recv_publish_msg.msg = p_src_data->client_recv_publish_msg.length ? (uint8_t *)osi_malloc(p_src_data->client_recv_publish_msg.length) : NULL;
+ if (p_dest_data->client_recv_publish_msg.ctx) {
+ memcpy(p_dest_data->client_recv_publish_msg.ctx, p_src_data->client_recv_publish_msg.ctx, sizeof(esp_ble_mesh_msg_ctx_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ if (p_src_data->client_recv_publish_msg.length) {
+ if (p_dest_data->client_recv_publish_msg.msg) {
+ memcpy(p_dest_data->client_recv_publish_msg.msg, p_src_data->client_recv_publish_msg.msg, p_src_data->client_recv_publish_msg.length);
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ }
+ break;
+ }
+ case ESP_BLE_MESH_MODEL_SEND_COMP_EVT: {
+ if (p_src_data->model_send_comp.ctx) {
+ p_dest_data->model_send_comp.ctx = osi_malloc(sizeof(esp_ble_mesh_msg_ctx_t));
+ if (p_dest_data->model_send_comp.ctx) {
+ memcpy(p_dest_data->model_send_comp.ctx, p_src_data->model_send_comp.ctx, sizeof(esp_ble_mesh_msg_ctx_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ }
+ case ESP_BLE_MESH_CLIENT_MODEL_SEND_TIMEOUT_EVT: {
+ if (p_src_data->client_send_timeout.ctx) {
+ p_dest_data->client_send_timeout.ctx = osi_malloc(sizeof(esp_ble_mesh_msg_ctx_t));
+ if (p_dest_data->client_send_timeout.ctx) {
+ memcpy(p_dest_data->client_send_timeout.ctx, p_src_data->client_send_timeout.ctx, sizeof(esp_ble_mesh_msg_ctx_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ }
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_free_req_data(btc_msg_t *msg)
+{
+ esp_ble_mesh_model_cb_param_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (esp_ble_mesh_model_cb_param_t *)(msg->arg);
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_MODEL_OPERATION_EVT: {
+ if (arg->model_operation.msg) {
+ osi_free(arg->model_operation.msg);
+ }
+ if (arg->model_operation.ctx) {
+ osi_free(arg->model_operation.ctx);
+ }
+ break;
+ }
+ case ESP_BLE_MESH_CLIENT_MODEL_RECV_PUBLISH_MSG_EVT: {
+ if (arg->client_recv_publish_msg.msg) {
+ osi_free(arg->client_recv_publish_msg.msg);
+ }
+ if (arg->client_recv_publish_msg.ctx) {
+ osi_free(arg->client_recv_publish_msg.ctx);
+ }
+ break;
+ }
+ case ESP_BLE_MESH_MODEL_SEND_COMP_EVT: {
+ if (arg->model_send_comp.ctx) {
+ osi_free(arg->model_send_comp.ctx);
+ }
+ break;
+ }
+ case ESP_BLE_MESH_CLIENT_MODEL_SEND_TIMEOUT_EVT: {
+ if (arg->client_send_timeout.ctx) {
+ osi_free(arg->client_send_timeout.ctx);
+ }
+ break;
+ }
+ default:
+ break;
+ }
+}
+
+static inline void btc_ble_mesh_cb_to_app(esp_ble_mesh_prov_cb_event_t event,
+ esp_ble_mesh_prov_cb_param_t *param)
+{
+ esp_ble_mesh_prov_cb_t btc_mesh_cb = (esp_ble_mesh_prov_cb_t)btc_profile_cb_get(BTC_PID_PROV);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+static inline void btc_ble_mesh_model_cb_to_app(esp_ble_mesh_model_cb_event_t event,
+ esp_ble_mesh_model_cb_param_t *param)
+{
+ esp_ble_mesh_model_cb_t btc_mesh_cb = (esp_ble_mesh_model_cb_t)btc_profile_cb_get(BTC_PID_MODEL);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+extern u32_t mesh_opcode;
+
+static void btc_ble_mesh_server_model_op_cb(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ esp_ble_mesh_model_cb_param_t mesh_param = {0};
+ mesh_param.model_operation.opcode = mesh_opcode;
+ mesh_param.model_operation.model = (esp_ble_mesh_model_t *)model;
+ mesh_param.model_operation.ctx = (esp_ble_mesh_msg_ctx_t *)ctx;
+ mesh_param.model_operation.length = buf->len;
+ mesh_param.model_operation.msg = buf->data;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_MODEL;
+ msg.act = ESP_BLE_MESH_MODEL_OPERATION_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_model_cb_param_t), btc_ble_mesh_copy_req_data);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+ return;
+}
+
+static void btc_ble_mesh_client_model_op_cb(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+ esp_ble_mesh_model_cb_param_t mesh_param = {0};
+ bt_mesh_client_common_t *client_op = NULL;
+ bt_mesh_internal_data_t *data = NULL;
+ bt_mesh_client_node_t *node = NULL;
+
+ if (!model || !model->user_data || !ctx || !buf) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ client_op = (bt_mesh_client_common_t *)model->user_data;
+ data = (bt_mesh_internal_data_t *)client_op->internal_data;
+
+ node = bt_mesh_is_model_message_publish(model, ctx, buf, false);
+ if (node == NULL) {
+ msg.act = ESP_BLE_MESH_CLIENT_MODEL_RECV_PUBLISH_MSG_EVT;
+ mesh_param.client_recv_publish_msg.opcode = mesh_opcode;
+ mesh_param.client_recv_publish_msg.model = (esp_ble_mesh_model_t *)model;
+ mesh_param.client_recv_publish_msg.ctx = (esp_ble_mesh_msg_ctx_t *)ctx;
+ mesh_param.client_recv_publish_msg.length = buf->len;
+ mesh_param.client_recv_publish_msg.msg = buf->data;
+ } else {
+ msg.act = ESP_BLE_MESH_MODEL_OPERATION_EVT;
+ mesh_param.model_operation.opcode = mesh_opcode;
+ mesh_param.model_operation.model = (esp_ble_mesh_model_t *)model;
+ mesh_param.model_operation.ctx = (esp_ble_mesh_msg_ctx_t *)ctx;
+ mesh_param.model_operation.length = buf->len;
+ mesh_param.model_operation.msg = buf->data;
+ }
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_MODEL;
+ if (msg.act != ESP_BLE_MESH_CLIENT_MODEL_RECV_PUBLISH_MSG_EVT) {
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&data->queue, node);
+ }
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_model_cb_param_t), btc_ble_mesh_copy_req_data);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+ return;
+}
+
+static void btc_ble_mesh_model_send_comp_cb(esp_ble_mesh_model_t *model, esp_ble_mesh_msg_ctx_t *ctx, u32_t opcode, int err)
+{
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ esp_ble_mesh_model_cb_param_t mesh_param = {0};
+ mesh_param.model_send_comp.err_code = err;
+ mesh_param.model_send_comp.opcode = opcode;
+ mesh_param.model_send_comp.model = model;
+ mesh_param.model_send_comp.ctx = ctx;
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_MODEL;
+ msg.act = ESP_BLE_MESH_MODEL_SEND_COMP_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_model_cb_param_t), btc_ble_mesh_copy_req_data);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+ return;
+}
+
+static void btc_ble_mesh_model_publish_comp_cb(esp_ble_mesh_model_t *model, int err)
+{
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ esp_ble_mesh_model_cb_param_t mesh_param = {0};
+ mesh_param.model_publish_comp.err_code = err;
+ mesh_param.model_publish_comp.model = model;
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_MODEL;
+ msg.act = ESP_BLE_MESH_MODEL_PUBLISH_COMP_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_model_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+ return;
+}
+
+#if defined(CONFIG_BT_MESH_NODE)
+static void btc_oob_pub_key_cb(void)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_OOB_PUB_KEY_EVT;
+
+ if (btc_transfer_context(&msg, NULL, 0, NULL) != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static int btc_output_number_cb(bt_mesh_output_action_t act, u32_t num)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ mesh_param.node_prov_output_num.action = (esp_ble_mesh_output_action_t)act;
+ mesh_param.node_prov_output_num.number = num;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_OUTPUT_NUMBER_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ return -1;
+ }
+
+ return 0;
+}
+
+static int btc_output_string_cb(const char *str)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ strncpy(mesh_param.node_prov_output_str.string, str, strlen(str));
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ return -1;
+ }
+
+ return 0;
+}
+
+static int btc_input_cb(bt_mesh_input_action_t act, u8_t size)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ mesh_param.node_prov_input.action = (esp_ble_mesh_input_action_t)act;
+ mesh_param.node_prov_input.size = size;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_INPUT_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ return -1;
+ }
+
+ return 0;
+}
+
+static void btc_link_open_cb(bt_mesh_prov_bearer_t bearer)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ mesh_param.node_prov_link_open.bearer = (esp_ble_mesh_prov_bearer_t)bearer;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static void btc_link_close_cb(bt_mesh_prov_bearer_t bearer)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ mesh_param.node_prov_link_close.bearer = (esp_ble_mesh_prov_bearer_t)bearer;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static void btc_complete_cb(u16_t net_idx, u16_t addr, u8_t flags, u32_t iv_index)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ mesh_param.node_prov_complete.net_idx = net_idx;
+ mesh_param.node_prov_complete.addr = addr;
+ mesh_param.node_prov_complete.flags = flags;
+ mesh_param.node_prov_complete.iv_index = iv_index;
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static void btc_reset_cb(void)
+{
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_NODE_PROV_RESET_EVT;
+ ret = btc_transfer_context(&msg, NULL, 0, NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+#endif /* defined(CONFIG_BT_MESH_NODE) */
+
+static void btc_prov_register_complete_cb(int err_code)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ mesh_param.prov_register_comp.err_code = err_code;
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROV_REGISTER_COMP_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static void btc_client_model_timeout_cb(struct k_work *work)
+{
+ bt_mesh_client_node_t *node = CONTAINER_OF(work,
+ bt_mesh_client_node_t,
+ timer.work);
+ __ASSERT(node, "unknown mesh client node.");
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+ bt_mesh_internal_data_t *data = NULL;
+ bt_mesh_client_common_t *client_op = node->ctx.model->user_data;
+ data = (bt_mesh_internal_data_t *)client_op->internal_data;
+ __ASSERT(client_op && data, "Invalid client user data or internal data.");
+ esp_ble_mesh_model_cb_param_t mesh_param = {0};
+ mesh_param.client_send_timeout.opcode = node->opcode;
+ mesh_param.client_send_timeout.model = (esp_ble_mesh_model_t *)node->ctx.model;
+ mesh_param.client_send_timeout.ctx = (esp_ble_mesh_msg_ctx_t *)&node->ctx;
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_MODEL;
+ msg.act = ESP_BLE_MESH_CLIENT_MODEL_SEND_TIMEOUT_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_model_cb_param_t), btc_ble_mesh_copy_req_data);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&data->queue, node);
+ return;
+}
+
+static int btc_model_publish_update(struct bt_mesh_model *mod)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ esp_ble_mesh_model_cb_param_t mesh_param = {0};
+ mesh_param.model_publish_update.model = (esp_ble_mesh_model_t *)mod;
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_MODEL;
+ msg.act = ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_model_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ return -1;
+ }
+
+ return 0;
+}
+
+static void btc_prov_set_complete_cb(esp_ble_mesh_prov_cb_param_t *param, uint8_t act)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = act;
+ ret = btc_transfer_context(&msg, param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+#if (CONFIG_BT_MESH_PROVISIONER)
+static void btc_provisioner_recv_unprov_adv_pkt_cb(const u8_t addr[6], const u8_t addr_type,
+ const u8_t adv_type, const u8_t dev_uuid[16],
+ u16_t oob_info, bt_mesh_prov_bearer_t bearer)
+{
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ LOG_DEBUG("%s", __func__);
+
+ memcpy(mesh_param.provisioner_recv_unprov_adv_pkt.dev_uuid, dev_uuid, 16);
+ memcpy(mesh_param.provisioner_recv_unprov_adv_pkt.addr, addr, ESP_BD_ADDR_LEN);
+ mesh_param.provisioner_recv_unprov_adv_pkt.addr_type = addr_type;
+ mesh_param.provisioner_recv_unprov_adv_pkt.oob_info = oob_info;
+ mesh_param.provisioner_recv_unprov_adv_pkt.adv_type = adv_type;
+ mesh_param.provisioner_recv_unprov_adv_pkt.bearer = bearer;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROVISIONER_RECV_UNPROV_ADV_PKT_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static int btc_provisioner_prov_read_oob_pub_key_cb(u8_t link_idx)
+{
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ LOG_DEBUG("%s", __func__);
+
+ mesh_param.provisioner_prov_read_oob_pub_key.link_idx = link_idx;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROVISIONER_PROV_READ_OOB_PUB_KEY_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ return -1;
+ }
+
+ return 0;
+}
+
+static int btc_provisioner_prov_input_cb(u8_t method, bt_mesh_output_action_t act, u8_t size, u8_t link_idx)
+{
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ LOG_DEBUG("%s", __func__);
+
+ mesh_param.provisioner_prov_input.method = (esp_ble_mesh_oob_method_t)method;
+ mesh_param.provisioner_prov_input.action = (esp_ble_mesh_output_action_t)act;
+ mesh_param.provisioner_prov_input.size = size;
+ mesh_param.provisioner_prov_input.link_idx = link_idx;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROVISIONER_PROV_INPUT_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ return -1;
+ }
+
+ return 0;
+}
+
+static int btc_provisioner_prov_output_cb(u8_t method, bt_mesh_input_action_t act, void *data, u8_t size, u8_t link_idx)
+{
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ LOG_DEBUG("%s", __func__);
+
+ mesh_param.provisioner_prov_output.method = (esp_ble_mesh_oob_method_t)method;
+ mesh_param.provisioner_prov_output.action = (esp_ble_mesh_input_action_t)act;
+ mesh_param.provisioner_prov_output.size = size;
+ mesh_param.provisioner_prov_output.link_idx = link_idx;
+ if (act == BT_MESH_ENTER_STRING) {
+ strncpy(mesh_param.provisioner_prov_output.string, (char *)data, size);
+ } else {
+ mesh_param.provisioner_prov_output.number = sys_get_le32((u8_t *)data);
+ }
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROVISIONER_PROV_OUTPUT_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ return -1;
+ }
+
+ return 0;
+}
+
+static void btc_provisioner_link_open_cb(bt_mesh_prov_bearer_t bearer)
+{
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ LOG_DEBUG("%s", __func__);
+
+ mesh_param.provisioner_prov_link_open.bearer = (esp_ble_mesh_prov_bearer_t)bearer;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROVISIONER_PROV_LINK_OPEN_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static void btc_provisioner_link_close_cb(bt_mesh_prov_bearer_t bearer, u8_t reason)
+{
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ LOG_DEBUG("%s", __func__);
+
+ mesh_param.provisioner_prov_link_close.bearer = (esp_ble_mesh_prov_bearer_t)bearer;
+ mesh_param.provisioner_prov_link_close.reason = reason;
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROVISIONER_PROV_LINK_CLOSE_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+
+static void btc_provisioner_prov_complete_cb(int node_idx, const u8_t device_uuid[16],
+ u16_t unicast_addr, u8_t element_num,
+ u16_t netkey_idx)
+{
+ LOG_DEBUG("%s", __func__);
+ btc_msg_t msg = {0};
+ bt_status_t ret;
+
+ esp_ble_mesh_prov_cb_param_t mesh_param = {0};
+ mesh_param.provisioner_prov_complete.node_idx = node_idx;
+ mesh_param.provisioner_prov_complete.unicast_addr = unicast_addr;
+ mesh_param.provisioner_prov_complete.element_num = element_num;
+ mesh_param.provisioner_prov_complete.netkey_idx = netkey_idx;
+ memcpy(mesh_param.provisioner_prov_complete.device_uuid, device_uuid, sizeof(esp_ble_mesh_octet16_t));
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_PROV;
+ msg.act = ESP_BLE_MESH_PROVISIONER_PROV_COMPLETE_EVT;
+ ret = btc_transfer_context(&msg, &mesh_param,
+ sizeof(esp_ble_mesh_prov_cb_param_t), NULL);
+
+ if (ret != BT_STATUS_SUCCESS) {
+ LOG_ERROR("%s btc_transfer_context failed\n", __func__);
+ }
+
+ return;
+}
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+int btc_ble_mesh_client_init(esp_ble_mesh_model_t *model)
+{
+ __ASSERT(model && model->op, "The client model or operation is invalid(NULL).");
+ esp_ble_mesh_model_op_t *op = model->op;
+ while (op != NULL && op->opcode != 0) {
+ op->param_cb = (esp_ble_mesh_cb_t)btc_ble_mesh_client_model_op_cb;
+ op++;
+ }
+ return bt_mesh_client_init((struct bt_mesh_model *)model);
+}
+
+int32_t btc_ble_mesh_model_pub_period_get(esp_ble_mesh_model_t *mod)
+{
+ return bt_mesh_model_pub_period_get((struct bt_mesh_model *)mod);
+}
+
+uint16_t btc_ble_mesh_get_primary_addr(void)
+{
+ return bt_mesh_primary_addr();
+}
+
+uint16_t *btc_ble_mesh_model_find_group(esp_ble_mesh_model_t *mod, uint16_t addr)
+{
+ return bt_mesh_model_find_group((struct bt_mesh_model *)mod, addr);
+}
+
+esp_ble_mesh_elem_t *btc_ble_mesh_elem_find(u16_t addr)
+{
+ return (esp_ble_mesh_elem_t *)bt_mesh_elem_find(addr);
+}
+
+uint8_t btc_ble_mesh_elem_count(void)
+{
+ return bt_mesh_elem_count();
+}
+
+esp_ble_mesh_model_t *btc_ble_mesh_model_find_vnd(esp_ble_mesh_elem_t *elem,
+ uint16_t company, uint16_t id)
+{
+ return (esp_ble_mesh_model_t *)bt_mesh_model_find_vnd((struct bt_mesh_elem *)elem, company, id);
+}
+
+esp_ble_mesh_model_t *btc_ble_mesh_model_find(esp_ble_mesh_elem_t *elem,
+ u16_t id)
+{
+ return (esp_ble_mesh_model_t *)bt_mesh_model_find((struct bt_mesh_elem *)elem, id);
+}
+
+const esp_ble_mesh_comp_t *btc_ble_mesh_comp_get(void)
+{
+ return (const esp_ble_mesh_comp_t *)bt_mesh_comp_get();
+}
+
+extern const struct bt_mesh_model_op bt_mesh_cfg_srv_op[];
+extern const struct bt_mesh_model_op bt_mesh_cfg_cli_op[];
+extern const struct bt_mesh_model_op bt_mesh_health_srv_op[];
+extern const struct bt_mesh_model_op bt_mesh_health_cli_op[];
+// For Generic Client
+extern const struct bt_mesh_model_op gen_onoff_cli_op[];
+extern const struct bt_mesh_model_op gen_level_cli_op[];
+extern const struct bt_mesh_model_op gen_def_trans_time_cli_op[];
+extern const struct bt_mesh_model_op gen_power_onoff_cli_op[];
+extern const struct bt_mesh_model_op gen_power_level_cli_op[];
+extern const struct bt_mesh_model_op gen_battery_cli_op[];
+extern const struct bt_mesh_model_op gen_location_cli_op[];
+extern const struct bt_mesh_model_op gen_property_cli_op[];
+// For Light Lightness CLient
+extern const struct bt_mesh_model_op light_lightness_cli_op[];
+extern const struct bt_mesh_model_op light_ctl_cli_op[];
+extern const struct bt_mesh_model_op light_hsl_cli_op[];
+extern const struct bt_mesh_model_op light_xyl_cli_op[];
+extern const struct bt_mesh_model_op light_lc_cli_op[];
+// For Sensor Client
+extern const struct bt_mesh_model_op sensor_cli_op[];
+// For Time Scene Client
+extern const struct bt_mesh_model_op time_cli_op[];
+extern const struct bt_mesh_model_op scene_cli_op[];
+extern const struct bt_mesh_model_op scheduler_cli_op[];
+
+static void btc_mesh_model_op_add(esp_ble_mesh_model_t *model)
+{
+ esp_ble_mesh_model_op_t *op = NULL;
+ esp_ble_mesh_model_pub_t *pub = NULL;
+
+ if (!model) {
+ return;
+ }
+
+ /* The vendor model opcode is 3 bytes. */
+ if ((model->op != NULL) && (model->op->opcode >= 0x10000)) {
+ goto add_model_op;
+ }
+
+ switch (model->model_id) {
+ case BT_MESH_MODEL_ID_CFG_SRV: {
+ model->op = (esp_ble_mesh_model_op_t *)bt_mesh_cfg_srv_op;
+ break;
+ }
+ case BT_MESH_MODEL_ID_CFG_CLI: {
+ model->op = (esp_ble_mesh_model_op_t *)bt_mesh_cfg_cli_op;
+ bt_mesh_config_client_t *cli = (bt_mesh_config_client_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_cfg_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_HEALTH_SRV: {
+ model->op = (esp_ble_mesh_model_op_t *)bt_mesh_health_srv_op;
+ break;
+ }
+ case BT_MESH_MODEL_ID_HEALTH_CLI: {
+ model->op = (esp_ble_mesh_model_op_t *)bt_mesh_health_cli_op;
+ bt_mesh_health_client_t *cli = (bt_mesh_health_client_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_health_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_ONOFF_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_onoff_cli_op);
+ bt_mesh_gen_onoff_cli_t *cli = (bt_mesh_gen_onoff_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_LEVEL_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_level_cli_op);
+ bt_mesh_gen_level_cli_t *cli = (bt_mesh_gen_level_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_def_trans_time_cli_op);
+ bt_mesh_gen_def_trans_time_cli_t *cli = (bt_mesh_gen_def_trans_time_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_POWER_ONOFF_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_power_onoff_cli_op);
+ bt_mesh_gen_power_onoff_cli_t *cli = (bt_mesh_gen_power_onoff_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_POWER_LEVEL_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_power_level_cli_op);
+ bt_mesh_gen_power_level_cli_t *cli = (bt_mesh_gen_power_level_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_BATTERY_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_battery_cli_op);
+ bt_mesh_gen_battery_cli_t *cli = (bt_mesh_gen_battery_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_LOCATION_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_location_cli_op);
+ bt_mesh_gen_location_cli_t *cli = (bt_mesh_gen_location_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_GEN_PROP_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)gen_property_cli_op);
+ bt_mesh_gen_property_cli_t *cli = (bt_mesh_gen_property_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_generic_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)light_lightness_cli_op);
+ bt_mesh_light_lightness_cli_t *cli = (bt_mesh_light_lightness_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_light_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_LIGHT_CTL_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)light_ctl_cli_op);
+ bt_mesh_light_ctl_cli_t *cli = (bt_mesh_light_ctl_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_light_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_LIGHT_HSL_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)light_hsl_cli_op);
+ bt_mesh_light_hsl_cli_t *cli = (bt_mesh_light_hsl_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_light_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_LIGHT_XYL_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)light_xyl_cli_op);
+ bt_mesh_light_xyl_cli_t *cli = (bt_mesh_light_xyl_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_light_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_LIGHT_LC_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)light_lc_cli_op);
+ bt_mesh_light_lc_cli_t *cli = (bt_mesh_light_lc_cli_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_light_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_SENSOR_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)sensor_cli_op);
+ bt_mesh_sensor_client_t *cli = (bt_mesh_sensor_client_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_sensor_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_TIME_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)time_cli_op);
+ bt_mesh_time_scene_client_t *cli = (bt_mesh_time_scene_client_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_time_scene_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_SCENE_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)scene_cli_op);
+ bt_mesh_time_scene_client_t *cli = (bt_mesh_time_scene_client_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_time_scene_client_publish_callback;
+ }
+ break;
+ }
+ case BT_MESH_MODEL_ID_SCHEDULER_CLI: {
+ model->op = ((esp_ble_mesh_model_op_t *)scheduler_cli_op);
+ bt_mesh_time_scene_client_t *cli = (bt_mesh_time_scene_client_t *)model->user_data;
+ if (cli != NULL) {
+ cli->publish_status = btc_mesh_time_scene_client_publish_callback;
+ }
+ break;
+ }
+ default: {
+ goto add_model_op;
+ }
+
+ }
+
+ return;
+
+add_model_op:
+ op = model->op;
+ pub = model->pub;
+ if (pub != NULL) {
+ pub->update = (esp_ble_mesh_cb_t)btc_model_publish_update;
+ }
+ while (op != NULL && op->opcode != 0) {
+ // if (op->min_len > BLE_MESH_MAX_DATA_SIZE || op->param_cb != 0) {
+ // --op;
+ // LOG_ERROR("%s, Invalid model operation min_len = %d, or param callbak = 0x%x, the last valid opcode = 0x%x",
+ // __func__, op->min_len, op->param_cb, op->opcode);
+ // return;
+ // }
+ op->param_cb = (esp_ble_mesh_cb_t)btc_ble_mesh_server_model_op_cb;
+ op++;
+ }
+
+ return;
+
+}
+
+void btc_mesh_prov_call_handler(btc_msg_t *msg)
+{
+ btc_ble_mesh_prov_args_t *arg = NULL;
+ esp_ble_mesh_prov_cb_param_t param = {0};
+ uint8_t act;
+
+ if (!msg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+ arg = (btc_ble_mesh_prov_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_APP_REGISTER: {
+ int err_code;
+ for (int i = 0; i < arg->mesh_reg.comp->element_count; i++) {
+ esp_ble_mesh_elem_t *elem = &arg->mesh_reg.comp->elements[i];
+ for (int j = 0; j < elem->sig_model_count; j++) {
+ esp_ble_mesh_model_t *models = &elem->sig_models[j];
+ /* The opcode of sig model should be 1 or 2 bytes. */
+ if ((models != NULL) && (models->op != NULL) && (models->op->opcode >= 0x10000)) {
+ err_code = -EINVAL;
+ btc_prov_register_complete_cb(err_code);
+ return;
+ }
+ btc_mesh_model_op_add(models);
+ }
+
+ for (int k = 0; k < elem->vnd_model_count; k++) {
+ esp_ble_mesh_model_t *vnd_models = &elem->vnd_models[k];
+ /* The opcode of vendor model should be 3 bytes. */
+ if (vnd_models->op->opcode < 0x10000) {
+ err_code = -EINVAL;
+ btc_prov_register_complete_cb(err_code);
+ return;
+ }
+ btc_mesh_model_op_add(vnd_models);
+ }
+ }
+#if CONFIG_BT_MESH_NODE
+ arg->mesh_reg.prov->oob_pub_key_cb = (esp_ble_mesh_cb_t)btc_oob_pub_key_cb;
+ arg->mesh_reg.prov->output_num_cb = (esp_ble_mesh_cb_t)btc_output_number_cb;
+ arg->mesh_reg.prov->output_str_cb = (esp_ble_mesh_cb_t)btc_output_string_cb;
+ arg->mesh_reg.prov->input_cb = (esp_ble_mesh_cb_t)btc_input_cb;
+ arg->mesh_reg.prov->link_open_cb = (esp_ble_mesh_cb_t)btc_link_open_cb;
+ arg->mesh_reg.prov->link_close_cb = (esp_ble_mesh_cb_t)btc_link_close_cb;
+ arg->mesh_reg.prov->complete_cb = (esp_ble_mesh_cb_t)btc_complete_cb;
+ arg->mesh_reg.prov->reset_cb = (esp_ble_mesh_cb_t)btc_reset_cb;
+#endif /* CONFIG_BT_MESH_NODE */
+#if (CONFIG_BT_MESH_PROVISIONER)
+ arg->mesh_reg.prov->provisioner_prov_read_oob_pub_key = (esp_ble_mesh_cb_t)btc_provisioner_prov_read_oob_pub_key_cb;
+ arg->mesh_reg.prov->provisioner_prov_input = (esp_ble_mesh_cb_t)btc_provisioner_prov_input_cb;
+ arg->mesh_reg.prov->provisioner_prov_output = (esp_ble_mesh_cb_t)btc_provisioner_prov_output_cb;
+ arg->mesh_reg.prov->provisioner_link_open = (esp_ble_mesh_cb_t)btc_provisioner_link_open_cb;
+ arg->mesh_reg.prov->provisioner_link_close = (esp_ble_mesh_cb_t)btc_provisioner_link_close_cb;
+ arg->mesh_reg.prov->provisioner_prov_comp = (esp_ble_mesh_cb_t)btc_provisioner_prov_complete_cb;
+ bt_mesh_prov_adv_pkt_cb_register(btc_provisioner_recv_unprov_adv_pkt_cb);
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+ err_code = bt_mesh_init((struct bt_mesh_prov *)arg->mesh_reg.prov,
+ (struct bt_mesh_comp *)arg->mesh_reg.comp);
+ // Don't forget to give the semaphore when ble mesh init finish.
+ xSemaphoreGive(arg->mesh_reg.semaphore);
+ btc_prov_register_complete_cb(err_code);
+ return;
+ }
+#if CONFIG_BT_MESH_NODE
+ case BTC_BLE_MESH_ACT_PROV_ENABLE:
+ LOG_DEBUG("%s, BTC_BLE_MESH_ACT_PROV_ENABLE, bearers = %d", __func__, arg->mesh_prov_enable.bearers);
+ act = ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT;
+ param.node_prov_enable_comp.err_code = bt_mesh_prov_enable(arg->mesh_prov_enable.bearers);
+ break;
+ case BTC_BLE_MESH_ACT_PROV_DISABLE:
+ LOG_DEBUG("%s, BTC_BLE_MESH_ACT_PROV_DISABLE, bearers = %d", __func__, arg->mesh_prov_disable.bearers);
+ act = ESP_BLE_MESH_NODE_PROV_DISABLE_COMP_EVT;
+ param.node_prov_disable_comp.err_code = bt_mesh_prov_disable(arg->mesh_prov_disable.bearers);
+ break;
+ case BTC_BLE_MESH_ACT_NODE_RESET:
+ LOG_DEBUG("%s, BTC_BLE_MESH_ACT_NODE_RESET", __func__);
+ bt_mesh_reset();
+ return;
+ case BTC_BLE_MESH_ACT_SET_OOB_PUB_KEY:
+ act = ESP_BLE_MESH_NODE_PROV_SET_OOB_PUB_KEY_COMP_EVT;
+ param.node_prov_set_oob_pub_key_comp.err_code =
+ bt_mesh_set_oob_pub_key(arg->set_oob_pub_key.pub_key_x,
+ arg->set_oob_pub_key.pub_key_y,
+ arg->set_oob_pub_key.private_key);
+ break;
+ case BTC_BLE_MESH_ACT_INPUT_NUMBER:
+ act = ESP_BLE_MESH_NODE_PROV_INPUT_NUMBER_COMP_EVT;
+ param.node_prov_input_num_comp.err_code = bt_mesh_input_number(arg->input_num.number);
+ break;
+ case BTC_BLE_MESH_ACT_INPUT_STRING:
+ act = ESP_BLE_MESH_NODE_PROV_INPUT_STRING_COMP_EVT;
+ param.node_prov_input_str_comp.err_code = bt_mesh_input_string(arg->input_str.string);
+ break;
+ case BTC_BLE_MESH_ACT_SET_DEVICE_NAME:
+ act = ESP_BLE_MESH_NODE_SET_UNPROV_DEV_NAME_COMP_EVT;
+ param.node_set_unprov_dev_name_comp.err_code = bt_mesh_set_device_name(arg->set_device_name.name);
+ break;
+ case BTC_BLE_MESH_ACT_PROXY_IDENTITY_ENABLE:
+ act = ESP_BLE_MESH_NODE_PROXY_IDENTITY_ENABLE_COMP_EVT;
+ param.node_proxy_identity_enable_comp.err_code = bt_mesh_proxy_identity_enable();
+ break;
+ case BTC_BLE_MESH_ACT_PROXY_GATT_ENABLE:
+ act = ESP_BLE_MESH_NODE_PROXY_GATT_ENABLE_COMP_EVT;
+ param.node_proxy_gatt_enable_comp.err_code = bt_mesh_proxy_gatt_enable();
+ break;
+ case BTC_BLE_MESH_ACT_PROXY_GATT_DISABLE:
+ act = ESP_BLE_MESH_NODE_PROXY_GATT_DISABLE_COMP_EVT;
+ param.node_proxy_gatt_disable_comp.err_code = bt_mesh_proxy_gatt_disable();
+ break;
+#endif /* CONFIG_BT_MESH_NODE */
+#if (CONFIG_BT_MESH_PROVISIONER)
+ case BTC_BLE_MESH_ACT_PROVISIONER_READ_OOB_PUB_KEY:
+ act = ESP_BLE_MESH_PROVISIONER_PROV_READ_OOB_PUB_KEY_COMP_EVT;
+ param.provisioner_prov_read_oob_pub_key_comp.err_code =
+ bt_mesh_prov_read_oob_pub_key(arg->provisioner_read_oob_pub_key.link_idx,
+ arg->provisioner_read_oob_pub_key.pub_key_x,
+ arg->provisioner_read_oob_pub_key.pub_key_y);
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_INPUT_STR:
+ act = ESP_BLE_MESH_PROVISIONER_PROV_INPUT_STRING_COMP_EVT;
+ param.provisioner_prov_input_str_comp.err_code =
+ bt_mesh_prov_set_oob_input_data((u8_t *)&arg->provisioner_input_str.string,
+ false, arg->provisioner_input_str.link_idx);
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_INPUT_NUM:
+ act = ESP_BLE_MESH_PROVISIONER_PROV_INPUT_NUMBER_COMP_EVT;
+ param.provisioner_prov_input_num_comp.err_code =
+ bt_mesh_prov_set_oob_input_data((u8_t *)&arg->provisioner_input_num.number,
+ true, arg->provisioner_input_num.link_idx);
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_ENABLE:
+ act = ESP_BLE_MESH_PROVISIONER_PROV_ENABLE_COMP_EVT;
+ param.provisioner_prov_enable_comp.err_code =
+ bt_mesh_provisioner_enable(arg->provisioner_enable.bearers);
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_DISABLE:
+ act = ESP_BLE_MESH_PROVISIONER_PROV_DISABLE_COMP_EVT;
+ param.provisioner_prov_disable_comp.err_code =
+ bt_mesh_provisioner_disable(arg->provisioner_disable.bearers);
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_DEV_ADD: {
+ struct bt_mesh_unprov_dev_add add_dev = {0};
+ add_dev.addr_type = arg->provisioner_dev_add.add_dev.addr_type;
+ add_dev.oob_info = arg->provisioner_dev_add.add_dev.oob_info;
+ add_dev.bearer = arg->provisioner_dev_add.add_dev.bearer;
+ memcpy(add_dev.addr, arg->provisioner_dev_add.add_dev.addr, 6);
+ memcpy(add_dev.uuid, arg->provisioner_dev_add.add_dev.uuid, 16);
+ act = ESP_BLE_MESH_PROVISIONER_ADD_UNPROV_DEV_COMP_EVT;
+ param.provisioner_add_unprov_dev_comp.err_code =
+ bt_mesh_provisioner_add_unprov_dev(&add_dev, arg->provisioner_dev_add.flags);
+ break;
+ }
+ case BTC_BLE_MESH_ACT_PROVISIONER_DEV_DEL: {
+ struct bt_mesh_device_delete del_dev = {0};
+ if (arg->provisioner_dev_del.del_dev.flag & DEL_DEV_ADDR_FLAG) {
+ del_dev.addr_type = arg->provisioner_dev_del.del_dev.addr_type;
+ memcpy(del_dev.addr, arg->provisioner_dev_del.del_dev.addr, 6);
+ } else if (arg->provisioner_dev_del.del_dev.flag & DEL_DEV_UUID_FLAG) {
+ memcpy(del_dev.uuid, arg->provisioner_dev_del.del_dev.uuid, 16);
+ }
+ act = ESP_BLE_MESH_PROVISIONER_DELETE_DEV_COMP_EVT;
+ param.provisioner_delete_dev_comp.err_code = bt_mesh_provisioner_delete_device(&del_dev);
+ break;
+ }
+ case BTC_BLE_MESH_ACT_PROVISIONER_SET_DEV_UUID_MATCH:
+ act = ESP_BLE_MESH_PROVISIONER_SET_DEV_UUID_MATCH_COMP_EVT;
+ param.provisioner_set_dev_uuid_match_comp.err_code =
+ bt_mesh_provisioner_set_dev_uuid_match(arg->set_dev_uuid_match.offset,
+ arg->set_dev_uuid_match.match_len,
+ arg->set_dev_uuid_match.match_val,
+ arg->set_dev_uuid_match.prov_after_match);
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_SET_PROV_DATA_INFO: {
+ struct bt_mesh_prov_data_info info = {0};
+ info.flag = arg->set_prov_data_info.prov_data.flag;
+ if (arg->set_prov_data_info.prov_data.flag & PROV_DATA_NET_IDX_FLAG) {
+ info.net_idx = arg->set_prov_data_info.prov_data.net_idx;
+ } else if (arg->set_prov_data_info.prov_data.flag & PROV_DATA_FLAGS_FLAG) {
+ info.flags = arg->set_prov_data_info.prov_data.flags;
+ } else if (arg->set_prov_data_info.prov_data.flag & PROV_DATA_IV_INDEX_FLAG) {
+ info.iv_index = arg->set_prov_data_info.prov_data.iv_index;
+ }
+ act = ESP_BLE_MESH_PROVISIONER_SET_PROV_DATA_INFO_COMP_EVT;
+ param.provisioner_set_prov_data_info_comp.err_code =
+ bt_mesh_provisioner_set_prov_data_info(&info);
+ break;
+ }
+ case BTC_BLE_MESH_ACT_PROVISIONER_SET_NODE_NAME:
+ act = ESP_BLE_MESH_PROVISIONER_SET_NODE_NAME_COMP_EVT;
+ param.provisioner_set_node_name_comp.err_code =
+ bt_mesh_provisioner_set_node_name(arg->set_node_name.index, arg->set_node_name.name);
+ if (param.provisioner_set_node_name_comp.err_code) {
+ param.provisioner_set_node_name_comp.node_index = ESP_BLE_MESH_INVALID_NODE_INDEX;
+ } else {
+ param.provisioner_set_node_name_comp.node_index = arg->set_node_name.index;
+ }
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_SET_LOCAL_APP_KEY: {
+ const u8_t *app_key = NULL;
+ const u8_t zero[16] = {0};
+ if (memcmp(arg->add_local_app_key.app_key, zero, 16)) {
+ app_key = arg->add_local_app_key.app_key;
+ }
+ act = ESP_BLE_MESH_PROVISIONER_ADD_LOCAL_APP_KEY_COMP_EVT;
+ param.provisioner_add_app_key_comp.err_code =
+ bt_mesh_provisioner_local_app_key_add(app_key, arg->add_local_app_key.net_idx,
+ &arg->add_local_app_key.app_idx);
+ if (param.provisioner_add_app_key_comp.err_code) {
+ param.provisioner_add_app_key_comp.app_idx = ESP_BLE_MESH_KEY_UNUSED;
+ } else {
+ param.provisioner_add_app_key_comp.app_idx = arg->add_local_app_key.app_idx;
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_PROVISIONER_BIND_LOCAL_MOD_APP:
+ act = ESP_BLE_MESH_PROVISIONER_BIND_APP_KEY_TO_MODEL_COMP_EVT;
+ param.provisioner_bind_app_key_to_model_comp.err_code =
+ bt_mesh_provisioner_bind_local_model_app_idx(arg->local_mod_app_bind.elem_addr,
+ arg->local_mod_app_bind.model_id,
+ arg->local_mod_app_bind.cid,
+ arg->local_mod_app_bind.app_idx);
+ break;
+ case BTC_BLE_MESH_ACT_PROVISIONER_ADD_LOCAL_NET_KEY: {
+ const u8_t *net_key = NULL;
+ const u8_t zero[16] = {0};
+ if (memcmp(arg->add_local_net_key.net_key, zero, 16)) {
+ net_key = arg->add_local_net_key.net_key;
+ }
+ act = ESP_BLE_MESH_PROVISIONER_ADD_LOCAL_NET_KEY_COMP_EVT;
+ param.provisioner_add_net_key_comp.err_code =
+ bt_mesh_provisioner_local_net_key_add(net_key, &arg->add_local_net_key.net_idx);
+ if (param.provisioner_add_net_key_comp.err_code) {
+ param.provisioner_add_net_key_comp.net_idx = ESP_BLE_MESH_KEY_UNUSED;
+ } else {
+ param.provisioner_add_net_key_comp.net_idx = arg->add_local_net_key.net_idx;
+ }
+ break;
+ }
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+#if (CONFIG_BT_MESH_FAST_PROV)
+ case BTC_BLE_MESH_ACT_SET_FAST_PROV_INFO:
+ act = ESP_BLE_MESH_SET_FAST_PROV_INFO_COMP_EVT;
+ param.set_fast_prov_info_comp.status_unicast =
+ bt_mesh_set_fast_prov_unicast_addr_range(arg->set_fast_prov_info.unicast_min,
+ arg->set_fast_prov_info.unicast_max);
+ param.set_fast_prov_info_comp.status_net_idx =
+ bt_mesh_set_fast_prov_net_idx(arg->set_fast_prov_info.net_idx);
+ bt_mesh_set_fast_prov_flags_iv_index(arg->set_fast_prov_info.flags,
+ arg->set_fast_prov_info.iv_index);
+ param.set_fast_prov_info_comp.status_match =
+ bt_mesh_provisioner_set_dev_uuid_match(arg->set_fast_prov_info.offset,
+ arg->set_fast_prov_info.match_len,
+ arg->set_fast_prov_info.match_val, false);
+ break;
+ case BTC_BLE_MESH_ACT_SET_FAST_PROV_ACTION:
+ act = ESP_BLE_MESH_SET_FAST_PROV_ACTION_COMP_EVT;
+ param.set_fast_prov_action_comp.status_action =
+ bt_mesh_set_fast_prov_action(arg->set_fast_prov_action.action);
+ break;
+#endif /* (CONFIG_BT_MESH_FAST_PROV) */
+ default:
+ LOG_WARN("%s, Invalid msg->act %d", __func__, msg->act);
+ return;
+ }
+
+ btc_prov_set_complete_cb(¶m, act);
+ return;
+}
+
+void btc_mesh_model_call_handler(btc_msg_t *msg)
+{
+ btc_ble_mesh_model_args_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_model_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_MODEL_PUBLISH: {
+ if (arg->model_publish.device_role == PROVISIONER) {
+ bt_mesh_role_param_t common = {0};
+ common.model = (struct bt_mesh_model *)(arg->model_publish.model);
+ common.role = arg->model_publish.device_role;
+ if (bt_mesh_copy_msg_role(&common)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ }
+ int err = bt_mesh_model_publish((struct bt_mesh_model *)arg->model_publish.model);
+ btc_ble_mesh_model_publish_comp_cb(arg->model_publish.model, err);
+ break;
+ }
+ case BTC_BLE_MESH_ACT_SERVER_MODEL_SEND: {
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(MIN(BT_MESH_TX_SDU_MAX, ESP_BLE_MESH_SDU_MAX_LEN));
+ net_buf_simple_init(msg, 0);
+ net_buf_simple_add_mem(msg, arg->model_send.data, arg->model_send.length);
+ arg->model_send.ctx->srv_send = true;
+ int err = bt_mesh_model_send((struct bt_mesh_model *)arg->model_send.model,
+ (struct bt_mesh_msg_ctx *)arg->model_send.ctx,
+ msg, NULL, NULL);
+ btc_ble_mesh_model_send_comp_cb(arg->model_send.model, arg->model_send.ctx,
+ arg->model_send.opcode, err);
+ break;
+ }
+ case BTC_BLE_MESH_ACT_CLIENT_MODEL_SEND: {
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(MIN(BT_MESH_TX_SDU_MAX, ESP_BLE_MESH_SDU_MAX_LEN));
+ bt_mesh_role_param_t common = {0};
+ net_buf_simple_init(msg, 0);
+ net_buf_simple_add_mem(msg, arg->model_send.data, arg->model_send.length);
+ arg->model_send.ctx->srv_send = false;
+ common.model = (struct bt_mesh_model *)(arg->model_send.model);
+ common.role = arg->model_send.device_role;
+ if (bt_mesh_copy_msg_role(&common)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ int err = bt_mesh_client_send_msg((struct bt_mesh_model *)arg->model_send.model,
+ arg->model_send.opcode,
+ (struct bt_mesh_msg_ctx *)arg->model_send.ctx, msg,
+ btc_client_model_timeout_cb, arg->model_send.msg_timeout,
+ arg->model_send.need_rsp, NULL, NULL);
+ btc_ble_mesh_model_send_comp_cb(arg->model_send.model, arg->model_send.ctx,
+ arg->model_send.opcode, err);
+ break;
+ }
+ default:
+ LOG_WARN("%s, Invalid msg->act %d", __func__, msg->act);
+ break;
+ }
+
+ btc_ble_mesh_prov_arg_deep_free(msg);
+}
+
+void btc_mesh_model_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_model_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_model_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_MODEL_EVT_MAX) {
+ btc_ble_mesh_model_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_free_req_data(msg);
+}
+
+void btc_mesh_prov_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_prov_cb_param_t *param = NULL;
+
+ if (!msg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_prov_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_PROV_EVT_MAX) {
+ btc_ble_mesh_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/btc/btc_ble_mesh_sensor_client.c b/components/bt/ble_mesh/btc/btc_ble_mesh_sensor_client.c
new file mode 100644
index 0000000000..448b5d0e89
--- /dev/null
+++ b/components/bt/ble_mesh/btc/btc_ble_mesh_sensor_client.c
@@ -0,0 +1,647 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "sensor_client.h"
+#include "btc_ble_mesh_sensor_client.h"
+#include "esp_ble_mesh_sensor_model_api.h"
+
+static inline void btc_ble_mesh_cb_to_app(esp_ble_mesh_sensor_client_cb_event_t event,
+ esp_ble_mesh_sensor_client_cb_param_t *param)
+{
+ esp_ble_mesh_sensor_client_cb_t btc_mesh_cb = (esp_ble_mesh_sensor_client_cb_t)btc_profile_cb_get(BTC_PID_SENSOR_CLIENT);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+void btc_ble_mesh_sensor_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ btc_ble_mesh_sensor_client_args_t *dst = (btc_ble_mesh_sensor_client_args_t *)p_dest;
+ btc_ble_mesh_sensor_client_args_t *src = (btc_ble_mesh_sensor_client_args_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !dst || !src) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_SENSOR_CLIENT_GET_STATE: {
+ dst->sensor_client_get_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->sensor_client_get_state.get_state = (esp_ble_mesh_sensor_client_get_state_t *)osi_malloc(sizeof(esp_ble_mesh_sensor_client_get_state_t));
+ if (dst->sensor_client_get_state.params && dst->sensor_client_get_state.get_state) {
+ memcpy(dst->sensor_client_get_state.params, src->sensor_client_get_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->sensor_client_get_state.get_state, src->sensor_client_get_state.get_state,
+ sizeof(esp_ble_mesh_sensor_client_get_state_t));
+
+ opcode = src->sensor_client_get_state.params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_GET:
+ if (src->sensor_client_get_state.get_state->column_get.raw_value_x) {
+ length = src->sensor_client_get_state.get_state->column_get.raw_value_x->len;
+ dst->sensor_client_get_state.get_state->column_get.raw_value_x = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_get_state.get_state->column_get.raw_value_x) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_get_state.get_state->column_get.raw_value_x, 0);
+ net_buf_simple_add_mem(dst->sensor_client_get_state.get_state->column_get.raw_value_x,
+ src->sensor_client_get_state.get_state->column_get.raw_value_x->data,
+ src->sensor_client_get_state.get_state->column_get.raw_value_x->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_GET:
+ if (src->sensor_client_get_state.get_state->series_get.raw_value_x1) {
+ length = src->sensor_client_get_state.get_state->series_get.raw_value_x1->len;
+ dst->sensor_client_get_state.get_state->series_get.raw_value_x1 = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_get_state.get_state->series_get.raw_value_x1) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_get_state.get_state->series_get.raw_value_x1, 0);
+ net_buf_simple_add_mem(dst->sensor_client_get_state.get_state->series_get.raw_value_x1,
+ src->sensor_client_get_state.get_state->series_get.raw_value_x1->data,
+ src->sensor_client_get_state.get_state->series_get.raw_value_x1->len);
+ }
+ if (src->sensor_client_get_state.get_state->series_get.raw_value_x2) {
+ length = src->sensor_client_get_state.get_state->series_get.raw_value_x2->len;
+ dst->sensor_client_get_state.get_state->series_get.raw_value_x2 = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_get_state.get_state->series_get.raw_value_x2) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_get_state.get_state->series_get.raw_value_x2, 0);
+ net_buf_simple_add_mem(dst->sensor_client_get_state.get_state->series_get.raw_value_x2,
+ src->sensor_client_get_state.get_state->series_get.raw_value_x2->data,
+ src->sensor_client_get_state.get_state->series_get.raw_value_x2->len);
+ }
+ break;
+ default:
+ break;
+ }
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_SENSOR_CLIENT_SET_STATE: {
+ dst->sensor_client_set_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->sensor_client_set_state.set_state = (esp_ble_mesh_sensor_client_set_state_t *)osi_malloc(sizeof(esp_ble_mesh_sensor_client_set_state_t));
+ if (dst->sensor_client_set_state.params && dst->sensor_client_set_state.set_state) {
+ memcpy(dst->sensor_client_set_state.params, src->sensor_client_set_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->sensor_client_set_state.set_state, src->sensor_client_set_state.set_state,
+ sizeof(esp_ble_mesh_sensor_client_set_state_t));
+
+ opcode = src->sensor_client_set_state.params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET:
+ if (src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down) {
+ length = src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down->len;
+ dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down, 0);
+ net_buf_simple_add_mem(dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down,
+ src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down->data,
+ src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down->len);
+ }
+ if (src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up) {
+ length = src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up->len;
+ dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up, 0);
+ net_buf_simple_add_mem(dst->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up,
+ src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up->data,
+ src->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up->len);
+ }
+ if (src->sensor_client_set_state.set_state->cadence_set.fast_cadence_low) {
+ length = src->sensor_client_set_state.set_state->cadence_set.fast_cadence_low->len;
+ dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_low = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_low) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_low, 0);
+ net_buf_simple_add_mem(dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_low,
+ src->sensor_client_set_state.set_state->cadence_set.fast_cadence_low->data,
+ src->sensor_client_set_state.set_state->cadence_set.fast_cadence_low->len);
+ }
+ if (src->sensor_client_set_state.set_state->cadence_set.fast_cadence_high) {
+ length = src->sensor_client_set_state.set_state->cadence_set.fast_cadence_high->len;
+ dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_high = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_high) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_high, 0);
+ net_buf_simple_add_mem(dst->sensor_client_set_state.set_state->cadence_set.fast_cadence_high,
+ src->sensor_client_set_state.set_state->cadence_set.fast_cadence_high->data,
+ src->sensor_client_set_state.set_state->cadence_set.fast_cadence_high->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET:
+ if (src->sensor_client_set_state.set_state->setting_set.sensor_setting_raw) {
+ length = src->sensor_client_set_state.set_state->setting_set.sensor_setting_raw->len;
+ dst->sensor_client_set_state.set_state->setting_set.sensor_setting_raw = bt_mesh_alloc_buf(length);
+ if (!dst->sensor_client_set_state.set_state->setting_set.sensor_setting_raw) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(dst->sensor_client_set_state.set_state->setting_set.sensor_setting_raw, 0);
+ net_buf_simple_add_mem(dst->sensor_client_set_state.set_state->setting_set.sensor_setting_raw,
+ src->sensor_client_set_state.set_state->setting_set.sensor_setting_raw->data,
+ src->sensor_client_set_state.set_state->setting_set.sensor_setting_raw->len);
+ }
+ break;
+ default:
+ break;
+ }
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ default:
+ LOG_DEBUG("%s Unhandled deep copy %d", __func__, msg->act);
+ break;
+ }
+}
+
+static void btc_ble_mesh_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ esp_ble_mesh_sensor_client_cb_param_t *p_dest_data = (esp_ble_mesh_sensor_client_cb_param_t *)p_dest;
+ esp_ble_mesh_sensor_client_cb_param_t *p_src_data = (esp_ble_mesh_sensor_client_cb_param_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !p_src_data || !p_dest_data) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_SENSOR_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_SENSOR_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_SENSOR_CLIENT_PUBLISH_EVT:
+ if (p_src_data->params) {
+ opcode = p_src_data->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS:
+ if (p_src_data->status_cb.descriptor_status.descriptor) {
+ length = p_src_data->status_cb.descriptor_status.descriptor->len;
+ p_dest_data->status_cb.descriptor_status.descriptor = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.descriptor_status.descriptor) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.descriptor_status.descriptor, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.descriptor_status.descriptor,
+ p_src_data->status_cb.descriptor_status.descriptor->data,
+ p_src_data->status_cb.descriptor_status.descriptor->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_STATUS:
+ if (p_src_data->status_cb.cadence_status.sensor_cadence_value) {
+ length = p_src_data->status_cb.cadence_status.sensor_cadence_value->len;
+ p_dest_data->status_cb.cadence_status.sensor_cadence_value = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.cadence_status.sensor_cadence_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.cadence_status.sensor_cadence_value, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.cadence_status.sensor_cadence_value,
+ p_src_data->status_cb.cadence_status.sensor_cadence_value->data,
+ p_src_data->status_cb.cadence_status.sensor_cadence_value->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS:
+ if (p_src_data->status_cb.settings_status.sensor_setting_property_ids) {
+ length = p_src_data->status_cb.settings_status.sensor_setting_property_ids->len;
+ p_dest_data->status_cb.settings_status.sensor_setting_property_ids = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.settings_status.sensor_setting_property_ids) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.settings_status.sensor_setting_property_ids, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.settings_status.sensor_setting_property_ids,
+ p_src_data->status_cb.settings_status.sensor_setting_property_ids->data,
+ p_src_data->status_cb.settings_status.sensor_setting_property_ids->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_STATUS:
+ if (p_src_data->status_cb.setting_status.sensor_setting_raw) {
+ length = p_src_data->status_cb.setting_status.sensor_setting_raw->len;
+ p_dest_data->status_cb.setting_status.sensor_setting_raw = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.setting_status.sensor_setting_raw) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.setting_status.sensor_setting_raw, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.setting_status.sensor_setting_raw,
+ p_src_data->status_cb.setting_status.sensor_setting_raw->data,
+ p_src_data->status_cb.setting_status.sensor_setting_raw->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_STATUS:
+ if (p_src_data->status_cb.sensor_status.marshalled_sensor_data) {
+ length = p_src_data->status_cb.sensor_status.marshalled_sensor_data->len;
+ p_dest_data->status_cb.sensor_status.marshalled_sensor_data = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.sensor_status.marshalled_sensor_data) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.sensor_status.marshalled_sensor_data, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.sensor_status.marshalled_sensor_data,
+ p_src_data->status_cb.sensor_status.marshalled_sensor_data->data,
+ p_src_data->status_cb.sensor_status.marshalled_sensor_data->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_STATUS:
+ if (p_src_data->status_cb.column_status.sensor_column_value) {
+ length = p_src_data->status_cb.column_status.sensor_column_value->len;
+ p_dest_data->status_cb.column_status.sensor_column_value = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.column_status.sensor_column_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.column_status.sensor_column_value, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.column_status.sensor_column_value,
+ p_src_data->status_cb.column_status.sensor_column_value->data,
+ p_src_data->status_cb.column_status.sensor_column_value->len);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_STATUS:
+ if (p_src_data->status_cb.series_status.sensor_series_value) {
+ length = p_src_data->status_cb.series_status.sensor_series_value->len;
+ p_dest_data->status_cb.series_status.sensor_series_value = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.series_status.sensor_series_value) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.series_status.sensor_series_value, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.series_status.sensor_series_value,
+ p_src_data->status_cb.series_status.sensor_series_value->data,
+ p_src_data->status_cb.series_status.sensor_series_value->len);
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_SENSOR_CLIENT_TIMEOUT_EVT:
+ if (p_src_data->params) {
+ p_dest_data->params = osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ if (p_dest_data->params) {
+ memcpy(p_dest_data->params, p_src_data->params, sizeof(esp_ble_mesh_client_common_param_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_free_req_data(btc_msg_t *msg)
+{
+ esp_ble_mesh_sensor_client_cb_param_t *arg = NULL;
+ u32_t opcode;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (esp_ble_mesh_sensor_client_cb_param_t *)(msg->arg);
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_SENSOR_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_SENSOR_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_SENSOR_CLIENT_PUBLISH_EVT:
+ if (arg->params) {
+ opcode = arg->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS:
+ bt_mesh_free_buf(arg->status_cb.descriptor_status.descriptor);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_STATUS:
+ bt_mesh_free_buf(arg->status_cb.cadence_status.sensor_cadence_value);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS:
+ bt_mesh_free_buf(arg->status_cb.settings_status.sensor_setting_property_ids);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_STATUS:
+ bt_mesh_free_buf(arg->status_cb.setting_status.sensor_setting_raw);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_STATUS:
+ bt_mesh_free_buf(arg->status_cb.sensor_status.marshalled_sensor_data);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_STATUS:
+ bt_mesh_free_buf(arg->status_cb.column_status.sensor_column_value);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_GET:
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_STATUS:
+ bt_mesh_free_buf(arg->status_cb.series_status.sensor_series_value);
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_SENSOR_CLIENT_TIMEOUT_EVT:
+ if (arg->params) {
+ osi_free(arg->params);
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+void btc_ble_mesh_sensor_client_arg_deep_free(btc_msg_t *msg)
+{
+ btc_ble_mesh_sensor_client_args_t *arg = NULL;
+ u32_t opcode = 0;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_sensor_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_SENSOR_CLIENT_GET_STATE:
+ if (arg->sensor_client_get_state.params) {
+ opcode = arg->sensor_client_get_state.params->opcode;
+ osi_free(arg->sensor_client_get_state.params);
+ }
+ if (arg->sensor_client_get_state.get_state) {
+ if (opcode) {
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_COLUMN_GET:
+ bt_mesh_free_buf(arg->sensor_client_get_state.get_state->column_get.raw_value_x);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SERIES_GET:
+ bt_mesh_free_buf(arg->sensor_client_get_state.get_state->series_get.raw_value_x1);
+ bt_mesh_free_buf(arg->sensor_client_get_state.get_state->series_get.raw_value_x2);
+ break;
+ default:
+ break;
+ }
+ }
+ osi_free(arg->sensor_client_get_state.get_state);
+ }
+ break;
+ case BTC_BLE_MESH_ACT_SENSOR_CLIENT_SET_STATE:
+ if (arg->sensor_client_set_state.params) {
+ opcode = arg->sensor_client_set_state.params->opcode;
+ osi_free(arg->sensor_client_set_state.params);
+ }
+ if (arg->sensor_client_set_state.set_state) {
+ if (opcode) {
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_CADENCE_SET:
+ bt_mesh_free_buf(arg->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_down);
+ bt_mesh_free_buf(arg->sensor_client_set_state.set_state->cadence_set.status_trigger_delta_up);
+ bt_mesh_free_buf(arg->sensor_client_set_state.set_state->cadence_set.fast_cadence_low);
+ bt_mesh_free_buf(arg->sensor_client_set_state.set_state->cadence_set.fast_cadence_high);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_SENSOR_SETTING_SET:
+ bt_mesh_free_buf(arg->sensor_client_set_state.set_state->setting_set.sensor_setting_raw);
+ break;
+ default:
+ break;
+ }
+ }
+ osi_free(arg->sensor_client_set_state.set_state);
+ }
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+static void btc_mesh_sensor_client_callback(esp_ble_mesh_sensor_client_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_SENSOR_CLIENT;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params,
+ sizeof(esp_ble_mesh_sensor_client_cb_param_t), btc_ble_mesh_copy_req_data);
+}
+
+void bt_mesh_callback_sensor_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len)
+{
+ esp_ble_mesh_sensor_client_cb_param_t cb_params = {0};
+ esp_ble_mesh_client_common_param_t params = {0};
+ size_t length;
+ uint8_t act;
+
+ if (!model || !ctx) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ switch (evt_type) {
+ case 0x00:
+ act = ESP_BLE_MESH_SENSOR_CLIENT_GET_STATE_EVT;
+ break;
+ case 0x01:
+ act = ESP_BLE_MESH_SENSOR_CLIENT_SET_STATE_EVT;
+ break;
+ case 0x02:
+ act = ESP_BLE_MESH_SENSOR_CLIENT_PUBLISH_EVT;
+ break;
+ case 0x03:
+ act = ESP_BLE_MESH_SENSOR_CLIENT_TIMEOUT_EVT;
+ break;
+ default:
+ LOG_ERROR("%s: unknown sensor status event type", __func__);
+ return;
+ }
+
+ params.opcode = opcode;
+ params.model = (esp_ble_mesh_model_t *)model;
+ params.ctx.net_idx = ctx->net_idx;
+ params.ctx.app_idx = ctx->app_idx;
+ params.ctx.addr = ctx->addr;
+ params.ctx.recv_ttl = ctx->recv_ttl;
+ params.ctx.recv_op = ctx->recv_op;
+ params.ctx.recv_dst = ctx->recv_dst;
+
+ cb_params.error_code = 0;
+ cb_params.params = ¶ms;
+
+ if (val && len) {
+ length = (len <= sizeof(cb_params.status_cb)) ? len : sizeof(cb_params.status_cb);
+ memcpy(&cb_params.status_cb, val, length);
+ }
+
+ btc_mesh_sensor_client_callback(&cb_params, act);
+}
+
+void btc_mesh_sensor_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf)
+{
+ if (!model || !ctx || !buf) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ bt_mesh_callback_sensor_status_to_btc(opcode, 0x02, model, ctx, buf->data, buf->len);
+}
+
+void btc_mesh_sensor_client_call_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_sensor_client_cb_param_t sensor_client_cb = {0};
+ esp_ble_mesh_client_common_param_t *params = NULL;
+ btc_ble_mesh_sensor_client_args_t *arg = NULL;
+ struct bt_mesh_common_param common = {0};
+ bt_mesh_role_param_t role_param = {0};
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_sensor_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_SENSOR_CLIENT_GET_STATE: {
+ params = arg->sensor_client_get_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ sensor_client_cb.params = arg->sensor_client_get_state.params;
+ sensor_client_cb.error_code =
+ bt_mesh_sensor_client_get_state(&common,
+ (void *)arg->sensor_client_get_state.get_state,
+ (void *)&sensor_client_cb.status_cb);
+ if (sensor_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_sensor_client_callback(&sensor_client_cb,
+ ESP_BLE_MESH_SENSOR_CLIENT_GET_STATE_EVT);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_SENSOR_CLIENT_SET_STATE: {
+ params = arg->sensor_client_set_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ sensor_client_cb.params = arg->sensor_client_set_state.params;
+ sensor_client_cb.error_code =
+ bt_mesh_sensor_client_set_state(&common,
+ (void *)arg->sensor_client_set_state.set_state,
+ (void *)&sensor_client_cb.status_cb);
+ if (sensor_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_sensor_client_callback(&sensor_client_cb,
+ ESP_BLE_MESH_SENSOR_CLIENT_SET_STATE_EVT);
+ }
+ break;
+ }
+ default:
+ break;
+ }
+
+ btc_ble_mesh_sensor_client_arg_deep_free(msg);
+}
+
+void btc_mesh_sensor_client_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_sensor_client_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_sensor_client_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_SENSOR_CLIENT_EVT_MAX) {
+ btc_ble_mesh_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_free_req_data(msg);
+}
+
diff --git a/components/bt/ble_mesh/btc/btc_ble_mesh_time_scene_client.c b/components/bt/ble_mesh/btc/btc_ble_mesh_time_scene_client.c
new file mode 100644
index 0000000000..48d03cb38a
--- /dev/null
+++ b/components/bt/ble_mesh/btc/btc_ble_mesh_time_scene_client.c
@@ -0,0 +1,384 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "btc/btc_manage.h"
+#include "osi/allocator.h"
+
+#include "time_scene_client.h"
+#include "btc_ble_mesh_time_scene_client.h"
+#include "esp_ble_mesh_time_scene_model_api.h"
+
+static inline void btc_ble_mesh_cb_to_app(esp_ble_mesh_time_scene_client_cb_event_t event,
+ esp_ble_mesh_time_scene_client_cb_param_t *param)
+{
+ esp_ble_mesh_time_scene_client_cb_t btc_mesh_cb = (esp_ble_mesh_time_scene_client_cb_t)btc_profile_cb_get(BTC_PID_TIME_SCENE_CLIENT);
+ if (btc_mesh_cb) {
+ btc_mesh_cb(event, param);
+ }
+}
+
+void btc_ble_mesh_time_scene_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ btc_ble_mesh_time_scene_client_args_t *dst = (btc_ble_mesh_time_scene_client_args_t *)p_dest;
+ btc_ble_mesh_time_scene_client_args_t *src = (btc_ble_mesh_time_scene_client_args_t *)p_src;
+
+ if (!msg || !dst || !src) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_GET_STATE: {
+ dst->time_scene_client_get_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->time_scene_client_get_state.get_state = (esp_ble_mesh_time_scene_client_get_state_t *)osi_malloc(sizeof(esp_ble_mesh_time_scene_client_get_state_t));
+ if (dst->time_scene_client_get_state.params && dst->time_scene_client_get_state.get_state) {
+ memcpy(dst->time_scene_client_get_state.params, src->time_scene_client_get_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->time_scene_client_get_state.get_state, src->time_scene_client_get_state.get_state,
+ sizeof(esp_ble_mesh_time_scene_client_get_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_SET_STATE: {
+ dst->time_scene_client_set_state.params = (esp_ble_mesh_client_common_param_t *)osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ dst->time_scene_client_set_state.set_state = (esp_ble_mesh_time_scene_client_set_state_t *)osi_malloc(sizeof(esp_ble_mesh_time_scene_client_set_state_t));
+ if (dst->time_scene_client_set_state.params && dst->time_scene_client_set_state.set_state) {
+ memcpy(dst->time_scene_client_set_state.params, src->time_scene_client_set_state.params,
+ sizeof(esp_ble_mesh_client_common_param_t));
+ memcpy(dst->time_scene_client_set_state.set_state, src->time_scene_client_set_state.set_state,
+ sizeof(esp_ble_mesh_time_scene_client_set_state_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ break;
+ }
+ default:
+ LOG_DEBUG("%s Unhandled deep copy %d", __func__, msg->act);
+ break;
+ }
+}
+
+static void btc_ble_mesh_copy_req_data(btc_msg_t *msg, void *p_dest, void *p_src)
+{
+ esp_ble_mesh_time_scene_client_cb_param_t *p_dest_data = (esp_ble_mesh_time_scene_client_cb_param_t *)p_dest;
+ esp_ble_mesh_time_scene_client_cb_param_t *p_src_data = (esp_ble_mesh_time_scene_client_cb_param_t *)p_src;
+ u32_t opcode;
+ u16_t length;
+
+ if (!msg || !p_src_data || !p_dest_data) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_PUBLISH_EVT:
+ if (p_src_data->params) {
+ opcode = p_src_data->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SCENE_STORE:
+ case ESP_BLE_MESH_MODEL_OP_SCENE_REGISTER_GET:
+ case ESP_BLE_MESH_MODEL_OP_SCENE_DELETE:
+ case ESP_BLE_MESH_MODEL_OP_SCENE_REGISTER_STATUS:
+ if (p_src_data->status_cb.scene_register_status.scenes) {
+ length = p_src_data->status_cb.scene_register_status.scenes->len;
+ p_dest_data->status_cb.scene_register_status.scenes = bt_mesh_alloc_buf(length);
+ if (!p_dest_data->status_cb.scene_register_status.scenes) {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ return;
+ }
+ net_buf_simple_init(p_dest_data->status_cb.scene_register_status.scenes, 0);
+ net_buf_simple_add_mem(p_dest_data->status_cb.scene_register_status.scenes,
+ p_src_data->status_cb.scene_register_status.scenes->data,
+ p_src_data->status_cb.scene_register_status.scenes->len);
+ }
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_TIMEOUT_EVT:
+ if (p_src_data->params) {
+ p_dest_data->params = osi_malloc(sizeof(esp_ble_mesh_client_common_param_t));
+ if (p_dest_data->params) {
+ memcpy(p_dest_data->params, p_src_data->params, sizeof(esp_ble_mesh_client_common_param_t));
+ } else {
+ LOG_ERROR("%s %d no mem", __func__, msg->act);
+ }
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+static void btc_ble_mesh_free_req_data(btc_msg_t *msg)
+{
+ esp_ble_mesh_time_scene_client_cb_param_t *arg = NULL;
+ u32_t opcode;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (esp_ble_mesh_time_scene_client_cb_param_t *)(msg->arg);
+
+ switch (msg->act) {
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_GET_STATE_EVT:
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_SET_STATE_EVT:
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_PUBLISH_EVT:
+ if (arg->params) {
+ opcode = arg->params->opcode;
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_SCENE_STORE:
+ case ESP_BLE_MESH_MODEL_OP_SCENE_REGISTER_GET:
+ case ESP_BLE_MESH_MODEL_OP_SCENE_DELETE:
+ case ESP_BLE_MESH_MODEL_OP_SCENE_REGISTER_STATUS:
+ bt_mesh_free_buf(arg->status_cb.scene_register_status.scenes);
+ break;
+ default:
+ break;
+ }
+ }
+ case ESP_BLE_MESH_TIME_SCENE_CLIENT_TIMEOUT_EVT:
+ if (arg->params) {
+ osi_free(arg->params);
+ }
+ break;
+ default:
+ break;
+ }
+}
+
+void btc_ble_mesh_time_scene_client_arg_deep_free(btc_msg_t *msg)
+{
+ btc_ble_mesh_time_scene_client_args_t *arg = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_time_scene_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_GET_STATE:
+ if (arg->time_scene_client_get_state.params) {
+ osi_free(arg->time_scene_client_get_state.params);
+ }
+ if (arg->time_scene_client_get_state.get_state) {
+ osi_free(arg->time_scene_client_get_state.get_state);
+ }
+ break;
+ case BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_SET_STATE:
+ if (arg->time_scene_client_set_state.params) {
+ osi_free(arg->time_scene_client_set_state.params);
+ }
+ if (arg->time_scene_client_set_state.set_state) {
+ osi_free(arg->time_scene_client_set_state.set_state);
+ }
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+static void btc_mesh_time_scene_client_callback(esp_ble_mesh_time_scene_client_cb_param_t *cb_params, uint8_t act)
+{
+ btc_msg_t msg = {0};
+
+ LOG_DEBUG("%s", __func__);
+
+ msg.sig = BTC_SIG_API_CB;
+ msg.pid = BTC_PID_TIME_SCENE_CLIENT;
+ msg.act = act;
+
+ btc_transfer_context(&msg, cb_params,
+ sizeof(esp_ble_mesh_time_scene_client_cb_param_t), btc_ble_mesh_copy_req_data);
+}
+
+void bt_mesh_callback_time_scene_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len)
+{
+ esp_ble_mesh_time_scene_client_cb_param_t cb_params = {0};
+ esp_ble_mesh_client_common_param_t params = {0};
+ size_t length;
+ uint8_t act;
+
+ if (!model || !ctx) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ switch (evt_type) {
+ case 0x00:
+ act = ESP_BLE_MESH_TIME_SCENE_CLIENT_GET_STATE_EVT;
+ break;
+ case 0x01:
+ act = ESP_BLE_MESH_TIME_SCENE_CLIENT_SET_STATE_EVT;
+ break;
+ case 0x02:
+ act = ESP_BLE_MESH_TIME_SCENE_CLIENT_PUBLISH_EVT;
+ break;
+ case 0x03:
+ act = ESP_BLE_MESH_TIME_SCENE_CLIENT_TIMEOUT_EVT;
+ break;
+ default:
+ LOG_ERROR("%s: unknown time scene status event type", __func__);
+ return;
+ }
+
+ params.opcode = opcode;
+ params.model = (esp_ble_mesh_model_t *)model;
+ params.ctx.net_idx = ctx->net_idx;
+ params.ctx.app_idx = ctx->app_idx;
+ params.ctx.addr = ctx->addr;
+ params.ctx.recv_ttl = ctx->recv_ttl;
+ params.ctx.recv_op = ctx->recv_op;
+ params.ctx.recv_dst = ctx->recv_dst;
+
+ cb_params.error_code = 0;
+ cb_params.params = ¶ms;
+
+ if (val && len) {
+ length = (len <= sizeof(cb_params.status_cb)) ? len : sizeof(cb_params.status_cb);
+ memcpy(&cb_params.status_cb, val, length);
+ }
+
+ btc_mesh_time_scene_client_callback(&cb_params, act);
+}
+
+void btc_mesh_time_scene_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf)
+{
+ if (!model || !ctx || !buf) {
+ LOG_ERROR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ bt_mesh_callback_time_scene_status_to_btc(opcode, 0x02, model, ctx, buf->data, buf->len);
+}
+
+void btc_mesh_time_scene_client_call_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_time_scene_client_cb_param_t time_scene_client_cb = {0};
+ btc_ble_mesh_time_scene_client_args_t *arg = NULL;
+ esp_ble_mesh_client_common_param_t *params = NULL;
+ struct bt_mesh_common_param common = {0};
+ bt_mesh_role_param_t role_param = {0};
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ arg = (btc_ble_mesh_time_scene_client_args_t *)(msg->arg);
+
+ switch (msg->act) {
+ case BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_GET_STATE: {
+ params = arg->time_scene_client_get_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ time_scene_client_cb.params = arg->time_scene_client_get_state.params;
+ time_scene_client_cb.error_code =
+ bt_mesh_time_scene_client_get_state(&common,
+ (void *)arg->time_scene_client_get_state.get_state,
+ (void *)&time_scene_client_cb.status_cb);
+ if (time_scene_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_time_scene_client_callback(&time_scene_client_cb,
+ ESP_BLE_MESH_TIME_SCENE_CLIENT_GET_STATE_EVT);
+ }
+ break;
+ }
+ case BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_SET_STATE: {
+ params = arg->time_scene_client_set_state.params;
+ role_param.model = (struct bt_mesh_model *)params->model;
+ role_param.role = params->msg_role;
+ if (bt_mesh_copy_msg_role(&role_param)) {
+ LOG_ERROR("%s: copy msg_role fail", __func__);
+ return;
+ }
+ common.opcode = params->opcode;
+ common.model = (struct bt_mesh_model *)params->model;
+ common.ctx.net_idx = params->ctx.net_idx;
+ common.ctx.app_idx = params->ctx.app_idx;
+ common.ctx.addr = params->ctx.addr;
+ common.ctx.send_rel = params->ctx.send_rel;
+ common.ctx.send_ttl = params->ctx.send_ttl;
+ common.msg_timeout = params->msg_timeout;
+
+ time_scene_client_cb.params = arg->time_scene_client_set_state.params;
+ time_scene_client_cb.error_code =
+ bt_mesh_time_scene_client_set_state(&common,
+ (void *)arg->time_scene_client_set_state.set_state,
+ (void *)&time_scene_client_cb.status_cb);
+ if (time_scene_client_cb.error_code) {
+ /* If send failed, callback error_code to app layer immediately */
+ btc_mesh_time_scene_client_callback(&time_scene_client_cb,
+ ESP_BLE_MESH_TIME_SCENE_CLIENT_SET_STATE_EVT);
+ }
+ break;
+ }
+ default:
+ break;
+ }
+
+ btc_ble_mesh_time_scene_client_arg_deep_free(msg);
+}
+
+void btc_mesh_time_scene_client_cb_handler(btc_msg_t *msg)
+{
+ esp_ble_mesh_time_scene_client_cb_param_t *param = NULL;
+
+ if (!msg || !msg->arg) {
+ LOG_ERROR("%s: parameter is NULL", __func__);
+ return;
+ }
+
+ param = (esp_ble_mesh_time_scene_client_cb_param_t *)(msg->arg);
+
+ if (msg->act < ESP_BLE_MESH_TIME_SCENE_CLIENT_EVT_MAX) {
+ btc_ble_mesh_cb_to_app(msg->act, param);
+ } else {
+ LOG_ERROR("%s, unknown msg->act = %d", __func__, msg->act);
+ }
+
+ btc_ble_mesh_free_req_data(msg);
+}
+
diff --git a/components/bt/ble_mesh/btc/include/btc_ble_mesh_config_client.h b/components/bt/ble_mesh/btc/include/btc_ble_mesh_config_client.h
new file mode 100644
index 0000000000..bb45221efd
--- /dev/null
+++ b/components/bt/ble_mesh/btc/include/btc_ble_mesh_config_client.h
@@ -0,0 +1,64 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BTC_BLE_MESH_CONFIG_CLIENT_H_
+#define _BTC_BLE_MESH_CONFIG_CLIENT_H_
+
+#include
+#include "btc/btc_task.h"
+#include "esp_ble_mesh_config_model_api.h"
+
+typedef enum {
+ BTC_BLE_MESH_ACT_CONFIG_CLIENT_GET_STATE,
+ BTC_BLE_MESH_ACT_CONFIG_CLIENT_SET_STATE,
+} btc_ble_mesh_cfg_client_act_t;
+
+typedef union {
+ struct ble_mesh_clg_client_get_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_cfg_client_get_state_t *get_state;
+ } cfg_client_get_state;
+ struct ble_mesh_clg_client_set_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_cfg_client_set_state_t *set_state;
+ } cfg_client_set_state;
+} btc_ble_mesh_cfg_client_args_t;
+
+void btc_mesh_cfg_client_call_handler(btc_msg_t *msg);
+
+void btc_mesh_cfg_client_cb_handler(btc_msg_t *msg);
+
+void btc_ble_mesh_cfg_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+int btc_ble_mesh_config_client_get_state(esp_ble_mesh_client_common_param_t *params, esp_ble_mesh_cfg_client_get_state_t *get_state,
+ esp_ble_mesh_cfg_client_cb_param_t *cfg_client_cb);
+
+int btc_ble_mesh_config_client_set_state(esp_ble_mesh_client_common_param_t *params, esp_ble_mesh_cfg_client_set_state_t *set_state,
+ esp_ble_mesh_cfg_client_cb_param_t *cfg_client_cb);
+
+void btc_mesh_cfg_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf);
+
+void bt_mesh_callback_config_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len);
+
+void btc_mesh_cfg_server_cb_handler(btc_msg_t *msg);
+
+void bt_mesh_callback_cfg_server_event_to_btc(u8_t evt_type, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len);
+
+#endif /* _BTC_BLE_MESH_CONFIG_CLIENT_H_ */
diff --git a/components/bt/ble_mesh/btc/include/btc_ble_mesh_generic_client.h b/components/bt/ble_mesh/btc/include/btc_ble_mesh_generic_client.h
new file mode 100644
index 0000000000..f2b8a3c217
--- /dev/null
+++ b/components/bt/ble_mesh/btc/include/btc_ble_mesh_generic_client.h
@@ -0,0 +1,51 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BTC_BLE_MESH_GENERIC_CLIENT_H_
+#define _BTC_BLE_MESH_GENERIC_CLIENT_H_
+
+#include
+#include "btc/btc_task.h"
+#include "esp_ble_mesh_generic_model_api.h"
+
+typedef enum {
+ BTC_BLE_MESH_ACT_GENERIC_CLIENT_GET_STATE,
+ BTC_BLE_MESH_ACT_GENERIC_CLIENT_SET_STATE,
+} btc_ble_mesh_generic_client_act_t;
+
+typedef union {
+ struct ble_mesh_generic_client_get_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_generic_client_get_state_t *get_state;
+ } generic_client_get_state;
+ struct ble_mesh_generic_client_set_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_generic_client_set_state_t *set_state;
+ } generic_client_set_state;
+} btc_ble_mesh_generic_client_args_t;
+
+void btc_mesh_generic_client_call_handler(btc_msg_t *msg);
+
+void btc_mesh_generic_client_cb_handler(btc_msg_t *msg);
+
+void btc_ble_mesh_generic_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+void btc_mesh_generic_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf);
+
+void bt_mesh_callback_generic_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len);
+#endif /* _BTC_BLE_MESH_GENERIC_CLIENT_H_ */
diff --git a/components/bt/ble_mesh/btc/include/btc_ble_mesh_health.h b/components/bt/ble_mesh/btc/include/btc_ble_mesh_health.h
new file mode 100644
index 0000000000..41010b935e
--- /dev/null
+++ b/components/bt/ble_mesh/btc/include/btc_ble_mesh_health.h
@@ -0,0 +1,79 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BTC_BLE_MESH_HEALTH_H_
+#define _BTC_BLE_MESH_HEALTH_H_
+
+#include
+#include "btc/btc_task.h"
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_health_model_api.h"
+
+typedef enum {
+ BTC_BLE_MESH_ACT_HEALTH_CLIENT_GET_STATE,
+ BTC_BLE_MESH_ACT_HEALTH_CLIENT_SET_STATE,
+ BTC_BLE_MESH_ACT_HEALTH_CLIENT_MAX,
+} btc_ble_mesh_health_client_act_t;
+
+typedef enum {
+ BTC_BLE_MESH_ACT_HEALTH_SERVER_FAULT_UPDATE,
+ BTC_BLE_MESH_ACT_HEALTH_SERVER_MAX,
+} btc_ble_mesh_health_server_act_t;
+
+typedef union {
+ struct ble_mesh_health_client_get_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_health_client_get_state_t *get_state;
+ } health_client_get_state;
+ struct ble_mesh_health_client_set_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_health_client_set_state_t *set_state;
+ } health_client_set_state;
+} btc_ble_mesh_health_client_args_t;
+
+typedef union {
+ struct ble_mesh_health_server_fault_update_args {
+ esp_ble_mesh_elem_t *element;
+ } fault_update;
+} btc_ble_mesh_health_server_args_t;
+
+void btc_mesh_health_client_call_handler(btc_msg_t *msg);
+
+void btc_mesh_health_client_cb_handler(btc_msg_t *msg);
+
+void btc_mesh_health_server_call_handler(btc_msg_t *msg);
+
+void btc_mesh_health_server_cb_handler(btc_msg_t *msg);
+
+void btc_ble_mesh_health_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+void btc_ble_mesh_health_server_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+int btc_ble_mesh_health_client_get_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_get_state_t *get_state,
+ esp_ble_mesh_health_client_cb_param_t *client_cb);
+
+int btc_ble_mesh_health_client_set_state(esp_ble_mesh_client_common_param_t *params,
+ esp_ble_mesh_health_client_set_state_t *set_state,
+ esp_ble_mesh_health_client_cb_param_t *client_cb);
+
+void btc_mesh_health_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf);
+
+void bt_mesh_callback_health_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, u16_t len);
+
+#endif /* _BTC_BLE_MESH_HEALTH_H_ */
diff --git a/components/bt/ble_mesh/btc/include/btc_ble_mesh_light_client.h b/components/bt/ble_mesh/btc/include/btc_ble_mesh_light_client.h
new file mode 100644
index 0000000000..7b07991796
--- /dev/null
+++ b/components/bt/ble_mesh/btc/include/btc_ble_mesh_light_client.h
@@ -0,0 +1,53 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BTC_BLE_MESH_LIGHT_CLIENT_H_
+#define _BTC_BLE_MESH_LIGHT_CLIENT_H_
+
+#include
+#include "btc/btc_task.h"
+#include "esp_ble_mesh_lighting_model_api.h"
+
+typedef enum {
+ BTC_BLE_MESH_ACT_LIGHT_CLIENT_GET_STATE,
+ BTC_BLE_MESH_ACT_LIGHT_CLIENT_SET_STATE,
+} btc_ble_mesh_light_client_act_t;
+
+typedef union {
+ struct ble_mesh_light_client_get_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_light_client_get_state_t *get_state;
+ } light_client_get_state;
+ struct ble_mesh_light_client_set_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_light_client_set_state_t *set_state;
+ } light_client_set_state;
+} btc_ble_mesh_light_client_args_t;
+
+void btc_mesh_light_client_call_handler(btc_msg_t *msg);
+
+void btc_mesh_light_client_cb_handler(btc_msg_t *msg);
+
+void btc_ble_mesh_light_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+void btc_mesh_light_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf);
+
+void bt_mesh_callback_light_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len);
+
+#endif /* _BTC_BLE_MESH_LIGHT_CLIENT_H_ */
+
diff --git a/components/bt/ble_mesh/btc/include/btc_ble_mesh_prov.h b/components/bt/ble_mesh/btc/include/btc_ble_mesh_prov.h
new file mode 100644
index 0000000000..c0b9b5646d
--- /dev/null
+++ b/components/bt/ble_mesh/btc/include/btc_ble_mesh_prov.h
@@ -0,0 +1,220 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BTC_BLE_MESH_PROV_H_
+#define _BTC_BLE_MESH_PROV_H_
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "freertos/queue.h"
+#include "freertos/semphr.h"
+
+#include "btc/btc_task.h"
+#include "esp_bt_defs.h"
+
+#include "mesh_access.h"
+#include "mesh_buf.h"
+#include "mesh_main.h"
+#include "esp_ble_mesh_defs.h"
+
+#if CONFIG_BT_MESH
+
+typedef enum {
+ BTC_BLE_MESH_ACT_APP_REGISTER = 0,
+ BTC_BLE_MESH_ACT_PROV_ENABLE,
+ BTC_BLE_MESH_ACT_PROV_DISABLE,
+ BTC_BLE_MESH_ACT_NODE_RESET,
+ BTC_BLE_MESH_ACT_SET_OOB_PUB_KEY,
+ BTC_BLE_MESH_ACT_INPUT_NUMBER,
+ BTC_BLE_MESH_ACT_INPUT_STRING,
+ BTC_BLE_MESH_ACT_SET_DEVICE_NAME,
+ BTC_BLE_MESH_ACT_PROXY_IDENTITY_ENABLE,
+ BTC_BLE_MESH_ACT_PROXY_GATT_ENABLE,
+ BTC_BLE_MESH_ACT_PROXY_GATT_DISABLE,
+#if (CONFIG_BT_MESH_PROVISIONER)
+ BTC_BLE_MESH_ACT_PROVISIONER_READ_OOB_PUB_KEY,
+ BTC_BLE_MESH_ACT_PROVISIONER_INPUT_STR,
+ BTC_BLE_MESH_ACT_PROVISIONER_INPUT_NUM,
+ BTC_BLE_MESH_ACT_PROVISIONER_ENABLE,
+ BTC_BLE_MESH_ACT_PROVISIONER_DISABLE,
+ BTC_BLE_MESH_ACT_PROVISIONER_DEV_ADD,
+ BTC_BLE_MESH_ACT_PROVISIONER_DEV_DEL,
+ BTC_BLE_MESH_ACT_PROVISIONER_SET_DEV_UUID_MATCH,
+ BTC_BLE_MESH_ACT_PROVISIONER_SET_PROV_DATA_INFO,
+ BTC_BLE_MESH_ACT_PROVISIONER_SET_NODE_NAME,
+ BTC_BLE_MESH_ACT_PROVISIONER_SET_LOCAL_APP_KEY,
+ BTC_BLE_MESH_ACT_PROVISIONER_BIND_LOCAL_MOD_APP,
+ BTC_BLE_MESH_ACT_PROVISIONER_ADD_LOCAL_NET_KEY,
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+ BTC_BLE_MESH_ACT_SET_FAST_PROV_INFO,
+ BTC_BLE_MESH_ACT_SET_FAST_PROV_ACTION,
+} btc_ble_mesh_prov_act_t;
+
+typedef enum {
+ BTC_BLE_MESH_ACT_MODEL_PUBLISH,
+ BTC_BLE_MESH_ACT_SERVER_MODEL_SEND,
+ BTC_BLE_MESH_ACT_CLIENT_MODEL_SEND
+} btc_ble_mesh_model_act_t;
+
+typedef union {
+ struct ble_mesh_reg_args {
+ esp_ble_mesh_prov_t *prov;
+ esp_ble_mesh_comp_t *comp;
+ xSemaphoreHandle semaphore;
+ uint8_t mesh_if;
+ } mesh_reg;
+ struct ble_mesh_prov_enable_args {
+ esp_ble_mesh_prov_bearer_t bearers;
+ } mesh_prov_enable;
+ struct ble_mesh_prov_disable_args {
+ esp_ble_mesh_prov_bearer_t bearers;
+ } mesh_prov_disable;
+ struct ble_mesh_set_oob_pub_key_args {
+ uint8_t pub_key_x[32];
+ uint8_t pub_key_y[32];
+ uint8_t private_key[32];
+ } set_oob_pub_key;
+ struct ble_mesh_input_num_args {
+ uint32_t number;
+ } input_num;
+ struct ble_mesh_input_str_args {
+ char string[8];
+ } input_str;
+ struct ble_mesh_set_device_name_args {
+ char name[ESP_BLE_MESH_DEVICE_NAME_MAX_LEN];
+ } set_device_name;
+#if (CONFIG_BT_MESH_PROVISIONER)
+ struct ble_mesh_provisioner_read_oob_pub_key_args {
+ uint8_t link_idx;
+ uint8_t pub_key_x[32];
+ uint8_t pub_key_y[32];
+ } provisioner_read_oob_pub_key;
+ struct ble_mesh_provisioner_input_str_args {
+ char string[8];
+ uint8_t link_idx;
+ } provisioner_input_str;
+ struct ble_mesh_provisioner_input_num_args {
+ uint32_t number;
+ uint8_t link_idx;
+ } provisioner_input_num;
+ struct ble_mesh_provisioner_enable_args {
+ esp_ble_mesh_prov_bearer_t bearers;
+ } provisioner_enable;
+ struct ble_mesh_provisioner_disable_args {
+ esp_ble_mesh_prov_bearer_t bearers;
+ } provisioner_disable;
+ struct ble_mesh_provisioner_dev_add_args {
+ esp_ble_mesh_unprov_dev_add_t add_dev;
+ esp_ble_mesh_dev_add_flag_t flags;
+ } provisioner_dev_add;
+ struct ble_mesh_provisioner_dev_del_args {
+ esp_ble_mesh_device_delete_t del_dev;
+ } provisioner_dev_del;
+ struct ble_mesh_provisioner_set_dev_uuid_match_args {
+ uint8_t offset;
+ uint8_t match_len;
+ uint8_t match_val[16];
+ bool prov_after_match;
+ } set_dev_uuid_match;
+ struct ble_mesh_provisioner_set_prov_net_idx_args {
+ esp_ble_mesh_prov_data_info_t prov_data;
+ } set_prov_data_info;
+ struct ble_mesh_provisioner_set_node_name_args {
+ int index;
+ char name[ESP_BLE_MESH_NODE_NAME_MAX_LEN];
+ } set_node_name;
+ struct ble_mesh_provisioner_add_local_app_key_args {
+ uint8_t app_key[16];
+ uint16_t net_idx;
+ uint16_t app_idx;
+ } add_local_app_key;
+ struct ble_mesh_provisioner_bind_local_mod_app_args {
+ uint16_t elem_addr;
+ uint16_t model_id;
+ uint16_t cid;
+ uint16_t app_idx;
+ } local_mod_app_bind;
+ struct ble_mesh_provisioner_add_local_net_key_args {
+ uint8_t net_key[16];
+ uint16_t net_idx;
+ } add_local_net_key;
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+#if (CONFIG_BT_MESH_FAST_PROV)
+ struct ble_mesh_set_fast_prov_info_args {
+ uint16_t unicast_min;
+ uint16_t unicast_max;
+ uint16_t net_idx;
+ uint8_t flags;
+ uint32_t iv_index;
+ uint8_t offset;
+ uint8_t match_len;
+ uint8_t match_val[16];
+ } set_fast_prov_info;
+ struct ble_mesh_set_fast_prov_action_args {
+ uint8_t action;
+ } set_fast_prov_action;
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+} btc_ble_mesh_prov_args_t;
+
+typedef union {
+ struct ble_mesh_model_publish_args {
+ esp_ble_mesh_model_t *model;
+ uint8_t device_role;
+ } model_publish;
+ struct ble_mesh_model_send_args {
+ esp_ble_mesh_model_t *model;
+ esp_ble_mesh_msg_ctx_t *ctx;
+ uint32_t opcode;
+ bool need_rsp;
+ uint16_t length;
+ uint8_t *data;
+ uint8_t device_role;
+ int32_t msg_timeout;
+ } model_send;
+} btc_ble_mesh_model_args_t;
+
+void btc_ble_mesh_prov_arg_deep_free(btc_msg_t *msg);
+
+void btc_ble_mesh_prov_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+int btc_ble_mesh_client_init(esp_ble_mesh_model_t *model);
+
+int32_t btc_ble_mesh_model_pub_period_get(esp_ble_mesh_model_t *mod);
+
+uint16_t btc_ble_mesh_get_primary_addr(void);
+
+uint16_t *btc_ble_mesh_model_find_group(esp_ble_mesh_model_t *mod, uint16_t addr);
+
+esp_ble_mesh_elem_t *btc_ble_mesh_elem_find(u16_t addr);
+
+uint8_t btc_ble_mesh_elem_count(void);
+
+esp_ble_mesh_model_t *btc_ble_mesh_model_find_vnd(esp_ble_mesh_elem_t *elem,
+ uint16_t company, uint16_t id);
+
+esp_ble_mesh_model_t *btc_ble_mesh_model_find(esp_ble_mesh_elem_t *elem,
+ u16_t id);
+
+const esp_ble_mesh_comp_t *btc_ble_mesh_comp_get(void);
+
+void btc_mesh_model_call_handler(btc_msg_t *msg);
+void btc_mesh_model_cb_handler(btc_msg_t *msg);
+
+void btc_mesh_prov_call_handler(btc_msg_t *msg);
+
+void btc_mesh_prov_cb_handler(btc_msg_t *msg);
+
+#endif /* _BTC_BLE_MESH_PROV_H_ */
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/btc/include/btc_ble_mesh_sensor_client.h b/components/bt/ble_mesh/btc/include/btc_ble_mesh_sensor_client.h
new file mode 100644
index 0000000000..af0524ec6f
--- /dev/null
+++ b/components/bt/ble_mesh/btc/include/btc_ble_mesh_sensor_client.h
@@ -0,0 +1,53 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BTC_BLE_MESH_SENSOR_CLIENT_H_
+#define _BTC_BLE_MESH_SENSOR_CLIENT_H_
+
+#include
+#include "btc/btc_task.h"
+#include "esp_ble_mesh_sensor_model_api.h"
+
+typedef enum {
+ BTC_BLE_MESH_ACT_SENSOR_CLIENT_GET_STATE,
+ BTC_BLE_MESH_ACT_SENSOR_CLIENT_SET_STATE,
+} btc_ble_mesh_sensor_client_act_t;
+
+typedef union {
+ struct ble_mesh_sensor_client_get_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_sensor_client_get_state_t *get_state;
+ } sensor_client_get_state;
+ struct ble_mesh_sensor_client_set_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_sensor_client_set_state_t *set_state;
+ } sensor_client_set_state;
+} btc_ble_mesh_sensor_client_args_t;
+
+void btc_mesh_sensor_client_call_handler(btc_msg_t *msg);
+
+void btc_mesh_sensor_client_cb_handler(btc_msg_t *msg);
+
+void btc_ble_mesh_sensor_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+void btc_mesh_sensor_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf);
+
+void bt_mesh_callback_sensor_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len);
+
+#endif /* _BTC_BLE_MESH_SENSOR_CLIENT_H_ */
+
diff --git a/components/bt/ble_mesh/btc/include/btc_ble_mesh_time_scene_client.h b/components/bt/ble_mesh/btc/include/btc_ble_mesh_time_scene_client.h
new file mode 100644
index 0000000000..aaf793a8bf
--- /dev/null
+++ b/components/bt/ble_mesh/btc/include/btc_ble_mesh_time_scene_client.h
@@ -0,0 +1,53 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BTC_BLE_MESH_TIME_SCENE_CLIENT_H_
+#define _BTC_BLE_MESH_TIME_SCENE_CLIENT_H_
+
+#include
+#include "btc/btc_task.h"
+#include "esp_ble_mesh_time_scene_model_api.h"
+
+typedef enum {
+ BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_GET_STATE,
+ BTC_BLE_MESH_ACT_TIME_SCENE_CLIENT_SET_STATE,
+} btc_ble_mesh_time_scene_client_act_t;
+
+typedef union {
+ struct ble_mesh_time_scene_client_get_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_time_scene_client_get_state_t *get_state;
+ } time_scene_client_get_state;
+ struct ble_mesh_time_scene_client_set_state_reg_args {
+ esp_ble_mesh_client_common_param_t *params;
+ esp_ble_mesh_time_scene_client_set_state_t *set_state;
+ } time_scene_client_set_state;
+} btc_ble_mesh_time_scene_client_args_t;
+
+void btc_mesh_time_scene_client_call_handler(btc_msg_t *msg);
+
+void btc_mesh_time_scene_client_cb_handler(btc_msg_t *msg);
+
+void btc_ble_mesh_time_scene_client_arg_deep_copy(btc_msg_t *msg, void *p_dest, void *p_src);
+
+void btc_mesh_time_scene_client_publish_callback(u32_t opcode, struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf);
+
+void bt_mesh_callback_time_scene_status_to_btc(u32_t opcode, u8_t evt_type,
+ struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ const u8_t *val, size_t len);
+
+#endif /* _BTC_BLE_MESH_TIME_SCENE_CLIENT_H_ */
+
diff --git a/components/bt/ble_mesh/mesh_core/Kconfig b/components/bt/ble_mesh/mesh_core/Kconfig
new file mode 100644
index 0000000000..b6175ead7c
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/Kconfig
@@ -0,0 +1,459 @@
+# Kconfig - Bluetooth Mesh configuration options
+
+#
+# Copyright (c) 2017 Intel Corporation
+#
+# SPDX-License-Identifier: Apache-2.0
+#
+
+menuconfig BT_MESH
+ bool "Bluetooth Mesh support"
+ select TINYCRYPT
+ select TINYCRYPT_AES
+ select TINYCRYPT_AES_CMAC
+ help
+ This option enables Bluetooth Mesh support. The specific
+ features that are available may depend on other features
+ that have been enabled in the stack, such as GATT support.
+
+if BT_MESH
+
+# Virtual option enabled whenever Generic Provisioning layer is needed
+config BT_MESH_PROV
+ bool
+
+config BT_MESH_PB_ADV
+ bool "Provisioning support using the advertising bearer (PB-ADV)"
+ select BT_MESH_PROV
+ default y
+ help
+ Enable this option to allow the device to be provisioned over
+ the advertising bearer.
+
+if BT_CONN
+
+# Virtual option enabled whenever any Proxy protocol is needed
+config BT_MESH_PROXY
+ bool
+
+config BT_MESH_PB_GATT
+ bool "Provisioning support using GATT (PB-GATT)"
+ select BT_MESH_PROXY
+ select BT_MESH_PROV
+ help
+ Enable this option to allow the device to be provisioned over
+ GATT.
+
+config BT_MESH_GATT_PROXY
+ bool "GATT Proxy Service"
+ select BT_MESH_PROXY
+ help
+ This option enables support for the Mesh GATT Proxy Service,
+ i.e. the ability to act as a proxy between a Mesh GATT Client
+ and a Mesh network.
+
+config BT_MESH_NODE_ID_TIMEOUT
+ int "Node Identity advertising timeout"
+ depends on BT_MESH_GATT_PROXY
+ range 1 60
+ default 60
+ help
+ This option determines for how long the local node advertises
+ using Node Identity. The given value is in seconds. The
+ specification limits this to 60 seconds, and implies that to
+ be the appropriate value as well, so just leaving this as the
+ default is the safest option.
+
+if BT_MESH_PROXY
+
+config BT_MESH_PROXY_FILTER_SIZE
+ int "Maximum number of filter entries per Proxy Client"
+ default 1
+ default 3 if BT_MESH_GATT_PROXY
+ range 1 32767
+ help
+ This option specifies how many Proxy Filter entries the local
+ node supports.
+
+endif # BT_MESH_PROXY
+
+endif # BT_CONN
+
+config BT_MESH_SELF_TEST
+ bool "Perform self-tests"
+ help
+ This option adds extra self-tests which are run every time
+ mesh networking is initialized.
+
+config BT_MESH_IV_UPDATE_TEST
+ bool "Test the IV Update Procedure"
+ help
+ This option removes the 96 hour limit of the IV Update
+ Procedure and lets the state be changed at any time.
+
+config BT_MESH_SUBNET_COUNT
+ int "Maximum number of mesh subnets per network"
+ default 1
+ range 1 4096
+ help
+ This option specifies how many subnets a Mesh network can
+ participate in at the same time.
+
+config BT_MESH_APP_KEY_COUNT
+ int "Maximum number of application keys per network"
+ default 1
+ range 1 4096
+ help
+ This option specifies how many application keys the device can
+ store per network.
+
+config BT_MESH_MODEL_KEY_COUNT
+ int "Maximum number of application keys per model"
+ default 1
+ range 1 4096
+ help
+ This option specifies how many application keys each model can
+ at most be bound to.
+
+config BT_MESH_MODEL_GROUP_COUNT
+ int "Maximum number of group address subscriptions per model"
+ default 1
+ range 1 4096
+ help
+ This option specifies how many group addresses each model can
+ at most be subscribed to.
+
+config BT_MESH_LABEL_COUNT
+ int "Maximum number of Label UUIDs used for Virtual Addresses"
+ default 1
+ range 0 4096
+ help
+ This option specifies how many Label UUIDs can be stored.
+
+config BT_MESH_CRPL
+ int "Maximum capacity of the replay protection list"
+ default 10
+ range 2 65535
+ help
+ This options specifies the maximum capacity of the replay
+ protection list. This option is similar to the network message
+ cache size, but has a different purpose.
+
+config BT_MESH_MSG_CACHE_SIZE
+ int "Network message cache size"
+ default 10
+ range 2 65535
+ help
+ Number of messages that are cached for the network. This helps
+ prevent unnecessary decryption operations and unnecessary
+ relays. This option is similar to the replay protection list,
+ but has a different purpose.
+
+config BT_MESH_ADV_BUF_COUNT
+ int "Number of advertising buffers"
+ default 6
+ range 6 256
+ help
+ Number of advertising buffers available. The transport layer
+ reserves ADV_BUF_COUNT - 3 buffers for outgoing segments. The
+ maximum outgoing SDU size is 12 times this number (out of which
+ 4 or 8 bytes is used for the Transport Layer MIC). For
+ example, 5 segments means the maximum SDU size is 60 bytes,
+ which leaves 56 bytes for application layer data using a
+ 4-byte MIC and 52 bytes using an 8-byte MIC.
+
+config BT_MESH_TX_SEG_MSG_COUNT
+ int "Maximum number of simultaneous outgoing segmented messages"
+ default 1
+ range 1 BT_MESH_ADV_BUF_COUNT
+ help
+ Maximum number of simultaneous outgoing multi-segment and/or
+ reliable messages.
+
+config BT_MESH_RX_SEG_MSG_COUNT
+ int "Maximum number of simultaneous incoming segmented messages"
+ default 1
+ range 1 255
+ help
+ Maximum number of simultaneous incoming multi-segment and/or
+ reliable messages.
+
+config BT_MESH_RX_SDU_MAX
+ int "Maximum incoming Upper Transport Access PDU length"
+ default 384
+ range 36 384
+ help
+ Maximum incoming Upper Transport Access PDU length. Leave this
+ to the default value, unless you really need to optimize memory
+ usage.
+
+config BT_MESH_RELAY
+ bool "Relay support"
+ help
+ Support for acting as a Mesh Relay Node.
+
+config BT_MESH_LOW_POWER
+ bool "Support for Low Power features"
+ help
+ Enable this option to be able to act as a Low Power Node.
+
+if BT_MESH_LOW_POWER
+
+config BT_MESH_LPN_ESTABLISHMENT
+ bool "Perform Friendship establishment using low power"
+ default y
+ help
+ Perform the Friendship establishment using low power, with
+ the help of a reduced scan duty cycle. The downside of this
+ is that the node may miss out on messages intended for it
+ until it has successfully set up Friendship with a Friend
+ node.
+
+config BT_MESH_LPN_AUTO
+ bool "Automatically start looking for Friend nodes once provisioned"
+ default y
+ help
+ Automatically enable LPN functionality once provisioned and start
+ looking for Friend nodes. If this option is disabled LPN mode
+ needs to be manually enabled by calling bt_mesh_lpn_set(true).
+
+config BT_MESH_LPN_AUTO_TIMEOUT
+ int "Time from last received message before going to LPN mode"
+ default 15
+ range 0 3600
+ depends on BT_MESH_LPN_AUTO
+ help
+ Time in seconds from the last received message, that the node
+ will wait before starting to look for Friend nodes.
+
+config BT_MESH_LPN_RETRY_TIMEOUT
+ int "Retry timeout for Friend requests"
+ default 8
+ range 1 3600
+ help
+ Time in seconds between Friend Requests, if a previous Friend
+ Request did not receive any acceptable Friend Offers.
+
+config BT_MESH_LPN_RSSI_FACTOR
+ int "RSSIFactor, used in the Friend Offer Delay calculation"
+ range 0 3
+ default 0
+ help
+ The contribution of the RSSI measured by the Friend node used
+ in Friend Offer Delay calculations. 0 = 1, 1 = 1.5, 2 = 2, 3 = 2.5.
+
+config BT_MESH_LPN_RECV_WIN_FACTOR
+ int "ReceiveWindowFactor, used in the Friend Offer Delay calculation"
+ range 0 3
+ default 0
+ help
+ The contribution of the supported Receive Window used in
+ Friend Offer Delay calculations. 0 = 1, 1 = 1.5, 2 = 2, 3 = 2.5.
+
+config BT_MESH_LPN_MIN_QUEUE_SIZE
+ int "Minimum size of acceptable friend queue (MinQueueSizeLog)"
+ range 1 7
+ default 1
+ help
+ The MinQueueSizeLog field is defined as log_2(N), where N is
+ the minimum number of maximum size Lower Transport PDUs that
+ the Friend node can store in its Friend Queue. As an example,
+ MinQueueSizeLog value 1 gives N = 2, and value 7 gives N = 128.
+
+config BT_MESH_LPN_RECV_DELAY
+ int "Receive delay requested by the local node"
+ range 10 255
+ default 100
+ help
+ The ReceiveDelay is the time between the Low Power node
+ sending a request and listening for a response. This delay
+ allows the Friend node time to prepare the response. The value
+ is in units of milliseconds.
+
+config BT_MESH_LPN_POLL_TIMEOUT
+ int "The value of the PollTimeout timer"
+ range 10 244735
+ default 300
+ help
+ PollTimeout timer is used to measure time between two
+ consecutive requests sent by the Low Power node. If no
+ requests are received by the Friend node before the
+ PollTimeout timer expires, then the friendship is considered
+ terminated. The value is in units of 100 milliseconds, so e.g.
+ a value of 300 means 30 seconds.
+
+config BT_MESH_LPN_INIT_POLL_TIMEOUT
+ int "The starting value of the PollTimeout timer"
+ range 10 BT_MESH_LPN_POLL_TIMEOUT
+ default BT_MESH_LPN_POLL_TIMEOUT
+ help
+ The initial value of the PollTimeout timer when Friendship
+ gets established for the first time. After this the timeout
+ will gradually grow toward the actual PollTimeout, doubling
+ in value for each iteration. The value is in units of 100
+ milliseconds, so e.g. a value of 300 means 30 seconds.
+
+config BT_MESH_LPN_SCAN_LATENCY
+ int "Latency for enabling scanning"
+ range 0 50
+ default 10
+ help
+ Latency in milliseconds that it takes to enable scanning. This
+ is in practice how much time in advance before the Receive Window
+ that scanning is requested to be enabled.
+
+config BT_MESH_LPN_GROUPS
+ int "Number of groups the LPN can subscribe to"
+ range 0 16384
+ default 8
+ help
+ Maximum number of groups that the LPN can subscribe to.
+endif # BT_MESH_LOW_POWER
+
+config BT_MESH_FRIEND
+ bool "Support for acting as a Friend Node"
+ help
+ Enable this option to be able to act as a Friend Node.
+
+if BT_MESH_FRIEND
+
+config BT_MESH_FRIEND_RECV_WIN
+ int "Friend Receive Window"
+ range 1 255
+ default 255
+ help
+ Receive Window in milliseconds supported by the Friend node.
+
+config BT_MESH_FRIEND_QUEUE_SIZE
+ int "Minimum number of buffers supported per Friend Queue"
+ range 2 65536
+ default 16
+ help
+ Minimum number of buffers available to be stored for each
+ local Friend Queue.
+
+config BT_MESH_FRIEND_SUB_LIST_SIZE
+ int "Friend Subscription List Size"
+ range 0 1023
+ default 3
+ help
+ Size of the Subscription List that can be supported by a
+ Friend node for a Low Power node.
+
+config BT_MESH_FRIEND_LPN_COUNT
+ int "Number of supported LPN nodes"
+ range 1 1000
+ default 2
+ help
+ Number of Low Power Nodes the Friend can have a Friendship
+ with simultaneously.
+
+config BT_MESH_FRIEND_SEG_RX
+ int "Number of incomplete segment lists per LPN"
+ range 1 1000
+ default 1
+ help
+ Number of incomplete segment lists that we track for each LPN
+ that we are Friends for. In other words, this determines how
+ many elements we can simultaneously be receiving segmented
+ messages from when the messages are going into the Friend queue.
+
+endif # BT_MESH_FRIEND
+
+config BT_MESH_CFG_CLI
+ bool "Support for Configuration Client Model"
+ help
+ Enable support for the configuration client model.
+
+config BT_MESH_HEALTH_CLI
+ bool "Support for Health Client Model"
+ help
+ Enable support for the health client model.
+
+config BT_MESH_SHELL
+ bool "Enable Bluetooth Mesh shell"
+ select CONSOLE_SHELL
+ depends on BT_MESH_CFG_CLI
+ depends on BT_MESH_HEALTH_CLI
+ help
+ Activate shell module that provides Bluetooth Mesh commands to
+ the console.
+
+config BT_MESH_DEBUG
+ bool "Enable debug logs"
+ depends on BT_DEBUG
+ help
+ Use this option to enable debug logs for the Bluetooth
+ Mesh functionality.
+
+if BT_MESH_DEBUG
+
+config BT_MESH_DEBUG_NET
+ bool "Network layer debug"
+ help
+ Use this option to enable Network layer debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_TRANS
+ bool "Transport layer debug"
+ help
+ Use this option to enable Transport layer debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_BEACON
+ bool "Beacon debug"
+ help
+ Use this option to enable Beacon-related debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_CRYPTO
+ bool "Crypto debug"
+ help
+ Use this option to enable cryptographic debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_PROV
+ bool "Provisioning debug"
+ help
+ Use this option to enable Provisioning debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_ACCESS
+ bool "Access layer debug"
+ help
+ Use this option to enable Access layer and device composition
+ related debug logs for Bluetooth Mesh.
+
+config BT_MESH_DEBUG_MODEL
+ bool "Foundation model debug"
+ help
+ Use this option to enable debug logs for the Foundation
+ Models.
+
+config BT_MESH_DEBUG_ADV
+ bool "Advertising debug"
+ help
+ Use this option to enable advertising debug logs for
+ the Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_LOW_POWER
+ bool "Low Power debug"
+ help
+ Use this option to enable Low Power debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_FRIEND
+ bool "Friend debug"
+ help
+ Use this option to enable Friend debug logs for the
+ Bluetooth Mesh functionality.
+
+config BT_MESH_DEBUG_PROXY
+ bool "Proxy debug"
+ depends on BT_MESH_PROXY
+ help
+ Use this option to enable Proxy protocol debug logs.
+
+endif # BT_MESH_DEBUG
+
+endif # BT_MESH
diff --git a/components/bt/ble_mesh/mesh_core/access.c b/components/bt/ble_mesh/mesh_core/access.c
new file mode 100644
index 0000000000..5003a199aa
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/access.c
@@ -0,0 +1,1036 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#include "errno.h"
+#include "mesh_util.h"
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+#include "mesh_trace.h"
+#include "mesh_kernel.h"
+#include "mesh_access.h"
+#include "mesh_main.h"
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_ACCESS)
+
+#include "mesh.h"
+#include "adv.h"
+#include "net.h"
+#include "lpn.h"
+#include "transport.h"
+#include "access.h"
+#include "foundation.h"
+
+#include "generic_client.h"
+#include "sensor_client.h"
+#include "time_scene_client.h"
+#include "light_client.h"
+
+#include "common.h"
+
+#define CODE_UNREACHABLE BT_ERR("The op_code not defined.");
+
+#define BT_MESH_SDU_MAX_LEN 384
+
+static const struct bt_mesh_comp *dev_comp;
+static u16_t dev_primary_addr;
+
+static const struct {
+ const u16_t id;
+ int (*const init)(struct bt_mesh_model *model, bool primary);
+} model_init[] = {
+ { BT_MESH_MODEL_ID_CFG_SRV, bt_mesh_cfg_srv_init },
+ { BT_MESH_MODEL_ID_HEALTH_SRV, bt_mesh_health_srv_init },
+#if defined(CONFIG_BT_MESH_CFG_CLI)
+ { BT_MESH_MODEL_ID_CFG_CLI, bt_mesh_cfg_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_HEALTH_CLI)
+ { BT_MESH_MODEL_ID_HEALTH_CLI, bt_mesh_health_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+ { BT_MESH_MODEL_ID_GEN_ONOFF_CLI, bt_mesh_gen_onoff_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_LEVEL_CLI)
+ { BT_MESH_MODEL_ID_GEN_LEVEL_CLI, bt_mesh_gen_level_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_DEF_TRANS_TIME_CLI)
+ { BT_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_CLI, bt_mesh_gen_def_trans_time_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_POWER_ONOFF_CLI)
+ { BT_MESH_MODEL_ID_GEN_POWER_ONOFF_CLI, bt_mesh_gen_pwr_onoff_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_POWER_LEVEL_CLI)
+ { BT_MESH_MODEL_ID_GEN_POWER_LEVEL_CLI, bt_mesh_gen_pwr_level_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_BATTERY_CLI)
+ { BT_MESH_MODEL_ID_GEN_BATTERY_CLI, bt_mesh_gen_battery_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_LOCATION_CLI)
+ { BT_MESH_MODEL_ID_GEN_LOCATION_CLI, bt_mesh_gen_location_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_GENERIC_PROPERTY_CLI)
+ { BT_MESH_MODEL_ID_GEN_PROP_CLI, bt_mesh_gen_property_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_SENSOR_CLI)
+ { BT_MESH_MODEL_ID_SENSOR_CLI, bt_mesh_sensor_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_TIME_CLI)
+ { BT_MESH_MODEL_ID_TIME_CLI, bt_mesh_time_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_SCENE_CLI)
+ { BT_MESH_MODEL_ID_SCENE_CLI, bt_mesh_scene_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_SCHEDULER_CLI)
+ { BT_MESH_MODEL_ID_SCHEDULER_CLI, bt_mesh_scheduler_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_LIGHT_LIGHTNESS_CLI)
+ { BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_CLI, bt_mesh_light_lightness_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_LIGHT_CTL_CLI)
+ { BT_MESH_MODEL_ID_LIGHT_CTL_CLI, bt_mesh_light_ctl_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_LIGHT_HSL_CLI)
+ { BT_MESH_MODEL_ID_LIGHT_HSL_CLI, bt_mesh_light_hsl_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_LIGHT_XYL_CLI)
+ { BT_MESH_MODEL_ID_LIGHT_XYL_CLI, bt_mesh_light_xyl_cli_init },
+#endif
+#if defined(CONFIG_BT_MESH_LIGHT_LC_CLI)
+ { BT_MESH_MODEL_ID_LIGHT_LC_CLI, bt_mesh_light_lc_cli_init },
+#endif
+};
+
+void bt_mesh_model_foreach(void (*func)(struct bt_mesh_model *mod,
+ struct bt_mesh_elem *elem,
+ bool vnd, bool primary,
+ void *user_data),
+ void *user_data)
+{
+ int i, j;
+
+ for (i = 0; i < dev_comp->elem_count; i++) {
+ struct bt_mesh_elem *elem = &dev_comp->elem[i];
+
+ for (j = 0; j < elem->model_count; j++) {
+ struct bt_mesh_model *model = &elem->models[j];
+
+ func(model, elem, false, i == 0, user_data);
+ }
+
+ for (j = 0; j < elem->vnd_model_count; j++) {
+ struct bt_mesh_model *model = &elem->vnd_models[j];
+
+ func(model, elem, true, i == 0, user_data);
+ }
+ }
+}
+
+s32_t bt_mesh_model_pub_period_get(struct bt_mesh_model *mod)
+{
+ int period = 0;
+
+ if (!mod->pub) {
+ BT_ERR("%s, model has no publication support", __func__);
+ return 0;
+ }
+
+ switch (mod->pub->period >> 6) {
+ case 0x00:
+ /* 1 step is 100 ms */
+ period = K_MSEC((mod->pub->period & BIT_MASK(6)) * 100);
+ break;
+ case 0x01:
+ /* 1 step is 1 second */
+ period = K_SECONDS(mod->pub->period & BIT_MASK(6));
+ break;
+ case 0x02:
+ /* 1 step is 10 seconds */
+ period = K_SECONDS((mod->pub->period & BIT_MASK(6)) * 10);
+ break;
+ case 0x03:
+ /* 1 step is 10 minutes */
+ period = K_MINUTES((mod->pub->period & BIT_MASK(6)) * 10);
+ break;
+ default:
+ CODE_UNREACHABLE;
+ }
+
+ return period >> mod->pub->period_div;
+}
+
+static s32_t next_period(struct bt_mesh_model *mod)
+{
+ struct bt_mesh_model_pub *pub = mod->pub;
+ u32_t elapsed, period;
+
+ if (!pub) {
+ BT_ERR("%s, model has no publication support", __func__);
+ return -ENOTSUP;
+ }
+
+ period = bt_mesh_model_pub_period_get(mod);
+ if (!period) {
+ return 0;
+ }
+
+ elapsed = k_uptime_get_32() - pub->period_start;
+
+ BT_DBG("Publishing took %ums", elapsed);
+
+ if (elapsed > period) {
+ BT_WARN("Publication sending took longer than the period");
+ /* Return smallest positive number since 0 means disabled */
+ return K_MSEC(1);
+ }
+
+ return period - elapsed;
+}
+
+static void publish_sent(int err, void *user_data)
+{
+ struct bt_mesh_model *mod = user_data;
+ s32_t delay;
+
+ BT_DBG("err %d", err);
+
+ if (!mod->pub) {
+ BT_ERR("%s, model has no publication support", __func__);
+ return;
+ }
+
+ if (mod->pub->count) {
+ delay = BT_MESH_PUB_TRANSMIT_INT(mod->pub->retransmit);
+ } else {
+ delay = next_period(mod);
+ }
+
+ if (delay) {
+ BT_DBG("Publishing next time in %dms", delay);
+ k_delayed_work_submit(&mod->pub->timer, delay);
+ }
+}
+
+static const struct bt_mesh_send_cb pub_sent_cb = {
+ .end = publish_sent,
+};
+
+static int publish_retransmit(struct bt_mesh_model *mod)
+{
+ struct net_buf_simple *sdu = NULL;
+ struct bt_mesh_model_pub *pub = mod->pub;
+ if (!pub) {
+ BT_ERR("%s, model has no publication support", __func__);
+ return -ENOTSUP;
+ }
+ struct bt_mesh_app_key *key;
+ struct bt_mesh_msg_ctx ctx = {
+ .addr = pub->addr,
+ .send_ttl = pub->ttl,
+ .model = mod,
+ .srv_send = (pub->dev_role == NODE ? true : false),
+ };
+ struct bt_mesh_net_tx tx = {
+ .ctx = &ctx,
+ .src = mod->elem->addr,
+ .xmit = bt_mesh_net_transmit_get(),
+ .friend_cred = pub->cred,
+ };
+ int err;
+
+ key = bt_mesh_app_key_find(pub->key);
+ if (!key) {
+ return -EADDRNOTAVAIL;
+ }
+
+ tx.sub = bt_mesh_subnet_get(key->net_idx);
+
+ ctx.net_idx = key->net_idx;
+ ctx.app_idx = key->app_idx;
+
+ sdu = bt_mesh_alloc_buf(min(BT_MESH_TX_SDU_MAX, BT_MESH_SDU_MAX_LEN));
+ if (!sdu) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+
+ net_buf_simple_init(sdu, 0);
+ net_buf_simple_add_mem(sdu, pub->msg->data, pub->msg->len);
+
+ pub->count--;
+
+ err = bt_mesh_trans_send(&tx, sdu, &pub_sent_cb, mod);
+
+ bt_mesh_free_buf(sdu);
+ return err;
+}
+
+static void mod_publish(struct k_work *work)
+{
+ struct bt_mesh_model_pub *pub = CONTAINER_OF(work,
+ struct bt_mesh_model_pub,
+ timer.work);
+ s32_t period_ms;
+ int err;
+
+ BT_DBG("%s", __func__);
+
+ period_ms = bt_mesh_model_pub_period_get(pub->mod);
+ BT_DBG("period %u ms", period_ms);
+
+ if (pub->count) {
+ err = publish_retransmit(pub->mod);
+ if (err) {
+ BT_ERR("Failed to retransmit (err %d)", err);
+
+ pub->count = 0;
+
+ /* Continue with normal publication */
+ if (period_ms) {
+ k_delayed_work_submit(&pub->timer, period_ms);
+ }
+ }
+
+ return;
+ }
+
+ if (!period_ms) {
+ return;
+ }
+
+ __ASSERT_NO_MSG(pub->update != NULL);
+
+ pub->period_start = k_uptime_get_32();
+
+ err = pub->update(pub->mod);
+ if (err) {
+ BT_ERR("Failed to update publication message");
+ return;
+ }
+
+ err = bt_mesh_model_publish(pub->mod);
+ if (err) {
+ BT_ERR("Publishing failed (err %d)", err);
+ }
+
+ if (pub->count) {
+ /* Retransmissions also control the timer */
+ k_delayed_work_cancel(&pub->timer);
+ }
+}
+
+static void mod_init(struct bt_mesh_model *mod, struct bt_mesh_elem *elem,
+ bool vnd, bool primary, void *user_data)
+{
+ int i;
+
+ mod->elem = elem;
+
+ if (mod->pub) {
+ mod->pub->mod = mod;
+ k_delayed_work_init(&mod->pub->timer, mod_publish);
+ }
+
+ for (i = 0; i < ARRAY_SIZE(mod->keys); i++) {
+ mod->keys[i] = BT_MESH_KEY_UNUSED;
+ }
+
+ mod->flags = 0;
+ mod->element_idx = elem - dev_comp->elem;
+ if (vnd) {
+ mod->model_idx = mod - elem->vnd_models;
+ } else {
+ mod->model_idx = mod - elem->models;
+ }
+
+ if (vnd) {
+ return;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(model_init); i++) {
+ BT_DBG("%s, model_init[i].id = %d, mod->id = %d", __func__, model_init[i].id, mod->id);
+ if (model_init[i].id == mod->id) {
+ model_init[i].init(mod, primary);
+ }
+ }
+}
+
+int bt_mesh_comp_register(const struct bt_mesh_comp *comp)
+{
+ /* There must be at least one element */
+ if (!comp->elem_count) {
+ return -EINVAL;
+ }
+
+ dev_comp = comp;
+
+ for (int i = 0; i < dev_comp->elem_count; i++) {
+ struct bt_mesh_elem *elem = &dev_comp->elem[i];
+
+ for (int j = 0; j < elem->model_count; j++) {
+ struct bt_mesh_model *model = &elem->models[j];
+ model->elem = elem;
+ }
+
+ for (int k = 0; k < elem->vnd_model_count; k++) {
+ struct bt_mesh_model *vnd_model = &elem->vnd_models[k];
+ vnd_model->elem = elem;
+ }
+ }
+
+ bt_mesh_model_foreach(mod_init, NULL);
+
+ return 0;
+}
+
+void bt_mesh_comp_provision(u16_t addr)
+{
+ int i;
+
+ dev_primary_addr = addr;
+
+ BT_DBG("addr 0x%04x elem_count %u", addr, dev_comp->elem_count);
+
+ for (i = 0; i < dev_comp->elem_count; i++) {
+ struct bt_mesh_elem *elem = &dev_comp->elem[i];
+
+ elem->addr = addr++;
+
+ BT_DBG("addr 0x%04x mod_count %u vnd_mod_count %u",
+ elem->addr, elem->model_count, elem->vnd_model_count);
+ }
+}
+
+void bt_mesh_comp_unprovision(void)
+{
+ BT_DBG("mesh unprovision complete, %s", __func__);
+
+ dev_primary_addr = BT_MESH_ADDR_UNASSIGNED;
+
+ bt_mesh_model_foreach(mod_init, NULL);
+}
+
+u16_t bt_mesh_primary_addr(void)
+{
+ return dev_primary_addr;
+}
+
+struct bt_mesh_model *bt_mesh_model_get(bool vnd, u8_t elem_idx, u8_t mod_idx)
+{
+ struct bt_mesh_elem *elem;
+
+ if (elem_idx >= dev_comp->elem_count) {
+ BT_ERR("Invalid element index %u", elem_idx);
+ return NULL;
+ }
+
+ elem = &dev_comp->elem[elem_idx];
+
+ if (vnd) {
+ if (mod_idx >= elem->vnd_model_count) {
+ BT_ERR("Invalid vendor model index %u", mod_idx);
+ return NULL;
+ }
+
+ return &elem->vnd_models[mod_idx];
+ } else {
+ if (mod_idx >= elem->model_count) {
+ BT_ERR("Invalid SIG model index %u", mod_idx);
+ return NULL;
+ }
+
+ return &elem->models[mod_idx];
+ }
+}
+
+
+u16_t *bt_mesh_model_find_group(struct bt_mesh_model *mod, u16_t addr)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(mod->groups); i++) {
+ if (mod->groups[i] == addr) {
+ return &mod->groups[i];
+ }
+ }
+
+ return NULL;
+}
+
+static struct bt_mesh_model *bt_mesh_elem_find_group(struct bt_mesh_elem *elem,
+ u16_t group_addr)
+{
+ struct bt_mesh_model *model;
+ u16_t *match;
+ int i;
+
+ for (i = 0; i < elem->model_count; i++) {
+ model = &elem->models[i];
+
+ match = bt_mesh_model_find_group(model, group_addr);
+ if (match) {
+ return model;
+ }
+ }
+
+ for (i = 0; i < elem->vnd_model_count; i++) {
+ model = &elem->vnd_models[i];
+
+ match = bt_mesh_model_find_group(model, group_addr);
+ if (match) {
+ return model;
+ }
+ }
+
+ return NULL;
+}
+
+struct bt_mesh_elem *bt_mesh_elem_find(u16_t addr)
+{
+ int i;
+
+ for (i = 0; i < dev_comp->elem_count; i++) {
+ struct bt_mesh_elem *elem = &dev_comp->elem[i];
+
+ if (BT_MESH_ADDR_IS_GROUP(addr) ||
+ BT_MESH_ADDR_IS_VIRTUAL(addr)) {
+ if (bt_mesh_elem_find_group(elem, addr)) {
+ return elem;
+ }
+ } else if (elem->addr == addr) {
+ return elem;
+ }
+ }
+
+ return NULL;
+}
+
+u8_t bt_mesh_elem_count(void)
+{
+ return dev_comp->elem_count;
+}
+
+static bool model_has_key(struct bt_mesh_model *mod, u16_t key)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(mod->keys); i++) {
+ if (mod->keys[i] == key) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+static const struct bt_mesh_model_op *find_op(struct bt_mesh_model *models,
+ u8_t model_count,
+ u16_t app_idx, u32_t opcode,
+ struct bt_mesh_model **model)
+{
+ u8_t i;
+
+ for (i = 0; i < model_count; i++) {
+ const struct bt_mesh_model_op *op;
+
+ *model = &models[i];
+
+ if (!model_has_key(*model, app_idx)) {
+ continue;
+ }
+
+ for (op = (*model)->op; op->func; op++) {
+ if (op->opcode == opcode) {
+ return op;
+ }
+ }
+ }
+
+ *model = NULL;
+ return NULL;
+}
+
+static int get_opcode(struct net_buf_simple *buf, u32_t *opcode)
+{
+ switch (buf->data[0] >> 6) {
+ case 0x00:
+ case 0x01:
+ if (buf->data[0] == 0x7f) {
+ BT_ERR("Ignoring RFU OpCode");
+ return -EINVAL;
+ }
+
+ *opcode = net_buf_simple_pull_u8(buf);
+ return 0;
+ case 0x02:
+ if (buf->len < 2) {
+ BT_ERR("Too short payload for 2-octet OpCode");
+ return -EINVAL;
+ }
+
+ *opcode = net_buf_simple_pull_be16(buf);
+ return 0;
+ case 0x03:
+ if (buf->len < 3) {
+ BT_ERR("Too short payload for 3-octet OpCode");
+ return -EINVAL;
+ }
+
+ *opcode = net_buf_simple_pull_u8(buf) << 16;
+ *opcode |= net_buf_simple_pull_le16(buf);
+ return 0;
+ }
+
+ /* Change by Espressif, we don't support the CODE_UNREACHABLE macro in ESP-IDF architecture */
+ return 0xffff;
+ //CODE_UNREACHABLE;
+}
+
+bool bt_mesh_fixed_group_match(u16_t addr)
+{
+ /* Check for fixed group addresses */
+ switch (addr) {
+ case BT_MESH_ADDR_ALL_NODES:
+ return true;
+ case BT_MESH_ADDR_PROXIES:
+ /* TODO: Proxy not yet supported */
+ return false;
+ case BT_MESH_ADDR_FRIENDS:
+ return (bt_mesh_friend_get() == BT_MESH_FRIEND_ENABLED);
+ case BT_MESH_ADDR_RELAYS:
+ return (bt_mesh_relay_get() == BT_MESH_RELAY_ENABLED);
+ default:
+ return false;
+ }
+}
+
+u32_t mesh_opcode;
+
+void bt_mesh_model_recv(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf)
+{
+ struct bt_mesh_model *models, *model;
+ const struct bt_mesh_model_op *op;
+ u32_t opcode;
+ u8_t count;
+ int i;
+
+ BT_DBG("app_idx 0x%04x src 0x%04x dst 0x%04x", rx->ctx.app_idx,
+ rx->ctx.addr, rx->dst);
+ BT_DBG("len %u: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (get_opcode(buf, &opcode) < 0) {
+ BT_WARN("Unable to decode OpCode");
+ return;
+ }
+
+ BT_DBG("OpCode 0x%08x", opcode);
+
+ mesh_opcode = opcode;
+ for (i = 0; i < dev_comp->elem_count; i++) {
+ struct bt_mesh_elem *elem = &dev_comp->elem[i];
+
+ if (BT_MESH_ADDR_IS_UNICAST(rx->dst)) {
+ if (elem->addr != rx->dst) {
+ continue;
+ }
+ } else if (BT_MESH_ADDR_IS_GROUP(rx->dst) ||
+ BT_MESH_ADDR_IS_VIRTUAL(rx->dst)) {
+ if (!bt_mesh_elem_find_group(elem, rx->dst)) {
+ continue;
+ }
+ } else if (i != 0 || !bt_mesh_fixed_group_match(rx->dst)) {
+ continue;
+ }
+
+ /* SIG models cannot contain 3-byte (vendor) OpCodes, and
+ * vendor models cannot contain SIG (1- or 2-byte) OpCodes, so
+ * the lookup must be done in one of the model lists.
+ */
+ if (opcode < 0x10000) {
+ models = elem->models;
+ count = elem->model_count;
+ } else {
+ models = elem->vnd_models;
+ count = elem->vnd_model_count;
+ }
+
+ op = find_op(models, count, rx->ctx.app_idx, opcode, &model);
+ if (op) {
+ struct net_buf_simple_state state;
+
+ if (buf->len < op->min_len) {
+ BT_ERR("Too short message for OpCode 0x%08x",
+ opcode);
+ continue;
+ }
+
+ /* The callback will likely parse the buffer, so
+ * store the parsing state in case multiple models
+ * receive the message.
+ */
+ net_buf_simple_save(buf, &state);
+ /** Change by Espressif, here we update recv_op with the
+ * value opcode got from the buf.
+ */
+ rx->ctx.recv_op = opcode;
+ /** Change by Espressif, Zephyr add 'recv_dst' in ctx in
+ * the lastest master branch, here we just update this
+ * value with rx->dst.
+ */
+ rx->ctx.recv_dst = rx->dst;
+ /** Change by Espressif, we update the model pointer to the
+ * found model when we received a message.
+ */
+ rx->ctx.model = model;
+ /** Change by Espressif, we update the srv_send flag to be
+ * true when we received a message. This flag will be used
+ * when a server model sends a status message and will
+ * have no impact on the client sent messages.
+ */
+ rx->ctx.srv_send = true;
+ op->func(model, &rx->ctx, buf);
+ net_buf_simple_restore(buf, &state);
+ } else {
+ BT_DBG("No OpCode 0x%08x for elem %d", opcode, i);
+ }
+ }
+}
+
+void bt_mesh_model_msg_init(struct net_buf_simple *msg, u32_t opcode)
+{
+ net_buf_simple_init(msg, 0);
+
+ if (opcode < 0x100) {
+ /* 1-byte OpCode */
+ net_buf_simple_add_u8(msg, opcode);
+ return;
+ }
+
+ if (opcode < 0x10000) {
+ /* 2-byte OpCode */
+ net_buf_simple_add_be16(msg, opcode);
+ return;
+ }
+
+ /* 3-byte OpCode */
+ net_buf_simple_add_u8(msg, ((opcode >> 16) & 0xff));
+ net_buf_simple_add_le16(msg, opcode & 0xffff);
+}
+
+static int model_send(struct bt_mesh_model *model,
+ struct bt_mesh_net_tx *tx, bool implicit_bind,
+ struct net_buf_simple *msg,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x dst 0x%04x", tx->ctx->net_idx,
+ tx->ctx->app_idx, tx->ctx->addr);
+ BT_DBG("len %u: %s", msg->len, bt_hex(msg->data, msg->len));
+
+ bool check = false;
+ u8_t role;
+
+ role = bt_mesh_get_msg_role(model, tx->ctx->srv_send);
+ if (role == ROLE_NVAL) {
+ BT_ERR("%s: get role fail", __func__);
+ return -EINVAL;
+ }
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (role == NODE) {
+ if (!bt_mesh_is_provisioned()) {
+ BT_ERR("%s: Local node is not yet provisioned", __func__);
+ return -EAGAIN;
+ }
+ if (!bt_mesh_is_provisioner_en()) {
+ check = true;
+ }
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == PROVISIONER) {
+ if (!provisioner_check_msg_dst_addr(tx->ctx->addr)) {
+ BT_ERR("%s: check msg dst_addr fail", __func__);
+ return -EINVAL;
+ }
+ if (bt_mesh_is_provisioner_en()) {
+ check = true;
+ }
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == PROVISIONER) {
+ if (!provisioner_check_msg_dst_addr(tx->ctx->addr)) {
+ BT_ERR("%s: check msg dst_addr fail", __func__);
+ return -EINVAL;
+ }
+ if (bt_mesh_is_provisioner_en()) {
+ check = true;
+ }
+ } else {
+ if (!bt_mesh_is_provisioned()) {
+ BT_ERR("Local node is not yet provisioned");
+ return -EAGAIN;
+ }
+ check = true;
+ }
+#endif
+
+ if (!check) {
+ BT_ERR("%s: check fail", __func__);
+ return -EINVAL;
+ }
+
+ if (net_buf_simple_tailroom(msg) < 4) {
+ BT_ERR("Not enough tailroom for TransMIC");
+ return -EINVAL;
+ }
+
+ if (msg->len > min(BT_MESH_TX_SDU_MAX, BT_MESH_SDU_MAX_LEN) - 4) {
+ BT_ERR("The message is too long");
+ return -EMSGSIZE;
+ }
+
+ if (!implicit_bind && !model_has_key(model, tx->ctx->app_idx)) {
+ BT_ERR("Model not bound to AppKey 0x%04x", tx->ctx->app_idx);
+ return -EINVAL;
+ }
+
+ return bt_mesh_trans_send(tx, msg, cb, cb_data);
+}
+
+int bt_mesh_model_send(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *msg,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ u8_t role;
+
+ role = bt_mesh_get_msg_role(model, ctx->srv_send);
+ if (role == ROLE_NVAL) {
+ BT_ERR("%s: get role fail", __func__);
+ return -EINVAL;
+ }
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (role == NODE) {
+ if (!bt_mesh_is_provisioner_en()) {
+ sub = bt_mesh_subnet_get(ctx->net_idx);
+ }
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ sub = provisioner_subnet_get(ctx->net_idx);
+ }
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == NODE) {
+ sub = bt_mesh_subnet_get(ctx->net_idx);
+ } else if (role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ sub = provisioner_subnet_get(ctx->net_idx);
+ }
+ } else if (role == FAST_PROV) {
+#if CONFIG_BT_MESH_FAST_PROV
+ sub = get_fast_prov_subnet(ctx->net_idx);
+#endif
+ }
+#endif
+
+ if (!sub) {
+ BT_ERR("%s: get sub fail", __func__);
+ return -EINVAL;
+ }
+
+ ctx->model = model;
+
+ struct bt_mesh_net_tx tx = {
+ .sub = sub,
+ .ctx = ctx,
+ .src = model->elem->addr,
+ .xmit = bt_mesh_net_transmit_get(),
+ .friend_cred = 0,
+ };
+
+ return model_send(model, &tx, false, msg, cb, cb_data);
+}
+
+int bt_mesh_model_publish(struct bt_mesh_model *model)
+{
+ struct net_buf_simple *sdu = NULL;
+ struct bt_mesh_model_pub *pub = model->pub;
+ struct bt_mesh_app_key *key = NULL;
+ struct bt_mesh_msg_ctx ctx = {0};
+ struct bt_mesh_net_tx tx = {
+ .sub = NULL,
+ .ctx = &ctx,
+ .src = model->elem->addr,
+ .xmit = bt_mesh_net_transmit_get(),
+ };
+ int err;
+
+ BT_DBG("%s", __func__);
+
+ if (!pub) {
+ BT_ERR("%s, model has no publication support", __func__);
+ return -ENOTSUP;
+ }
+
+ if (pub->addr == BT_MESH_ADDR_UNASSIGNED) {
+ return -EADDRNOTAVAIL;
+ }
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (pub->dev_role == NODE) {
+ if (bt_mesh_is_provisioned()) {
+ key = bt_mesh_app_key_find(pub->key);
+ }
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (pub->dev_role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ key = provisioner_app_key_find(pub->key);
+ }
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (pub->dev_role == NODE) {
+ if (bt_mesh_is_provisioned()) {
+ key = bt_mesh_app_key_find(pub->key);
+ }
+ } else if (pub->dev_role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ key = provisioner_app_key_find(pub->key);
+ }
+ }
+#endif
+
+ if (!key) {
+ BT_ERR("%s: get app key fail", __func__);
+ return -EADDRNOTAVAIL;
+ }
+
+ if (pub->msg->len + 4 > min(BT_MESH_TX_SDU_MAX, BT_MESH_SDU_MAX_LEN)) {
+ BT_ERR("Message does not fit maximum SDU size");
+ return -EMSGSIZE;
+ }
+
+ if (pub->count) {
+ BT_WARN("Clearing publish retransmit timer");
+ k_delayed_work_cancel(&pub->timer);
+ }
+
+ sdu = bt_mesh_alloc_buf(min(BT_MESH_TX_SDU_MAX, BT_MESH_SDU_MAX_LEN));
+ if (!sdu) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+
+ net_buf_simple_init(sdu, 0);
+ net_buf_simple_add_mem(sdu, pub->msg->data, pub->msg->len);
+
+ ctx.addr = pub->addr;
+ ctx.send_ttl = pub->ttl;
+ ctx.net_idx = key->net_idx;
+ ctx.app_idx = key->app_idx;
+ ctx.srv_send = pub->dev_role == NODE ? true : false;
+
+ tx.friend_cred = pub->cred;
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (pub->dev_role == NODE) {
+ if (bt_mesh_is_provisioned()) {
+ tx.sub = bt_mesh_subnet_get(ctx.net_idx);
+ }
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (pub->dev_role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ tx.sub = provisioner_subnet_get(ctx.net_idx);
+ }
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (pub->dev_role == NODE) {
+ if (bt_mesh_is_provisioned()) {
+ tx.sub = bt_mesh_subnet_get(ctx.net_idx);
+ }
+ } else if (pub->dev_role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ tx.sub = provisioner_subnet_get(ctx.net_idx);
+ }
+ }
+#endif
+
+ if (!tx.sub) {
+ BT_ERR("%s: Failed to get subnet", __func__);
+ bt_mesh_free_buf(sdu);
+ return -EADDRNOTAVAIL;
+ }
+
+ pub->count = BT_MESH_PUB_TRANSMIT_COUNT(pub->retransmit);
+
+ BT_DBG("Publish Retransmit Count %u Interval %ums", pub->count,
+ BT_MESH_PUB_TRANSMIT_INT(pub->retransmit));
+
+ err = model_send(model, &tx, true, sdu, &pub_sent_cb, model);
+ if (err) {
+ pub->count = 0;
+ }
+
+ bt_mesh_free_buf(sdu);
+ return err;
+}
+
+struct bt_mesh_model *bt_mesh_model_find_vnd(struct bt_mesh_elem *elem,
+ u16_t company, u16_t id)
+{
+ u8_t i;
+
+ for (i = 0; i < elem->vnd_model_count; i++) {
+ if (elem->vnd_models[i].vnd.company == company &&
+ elem->vnd_models[i].vnd.id == id) {
+ return &elem->vnd_models[i];
+ }
+ }
+
+ return NULL;
+}
+
+struct bt_mesh_model *bt_mesh_model_find(struct bt_mesh_elem *elem,
+ u16_t id)
+{
+ u8_t i;
+
+ for (i = 0; i < elem->model_count; i++) {
+ if (elem->models[i].id == id) {
+ return &elem->models[i];
+ }
+ }
+
+ return NULL;
+}
+
+const struct bt_mesh_comp *bt_mesh_comp_get(void)
+{
+ return dev_comp;
+}
+#endif /* CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/access.h b/components/bt/ble_mesh/mesh_core/access.h
new file mode 100644
index 0000000000..3a45e49b71
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/access.h
@@ -0,0 +1,62 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef _ACCESS_H_
+#define _ACCESS_H_
+
+#include "mesh_access.h"
+#include "mesh_buf.h"
+#include "net.h"
+
+#if CONFIG_BT_MESH
+
+/* bt_mesh_model.flags */
+enum {
+ BT_MESH_MOD_BIND_PENDING = BIT(0),
+ BT_MESH_MOD_SUB_PENDING = BIT(1),
+ BT_MESH_MOD_PUB_PENDING = BIT(2),
+};
+
+void bt_mesh_elem_register(struct bt_mesh_elem *elem, u8_t count);
+
+u8_t bt_mesh_elem_count(void);
+
+/* Find local element based on unicast or group address */
+struct bt_mesh_elem *bt_mesh_elem_find(u16_t addr);
+
+struct bt_mesh_model *bt_mesh_model_find_vnd(struct bt_mesh_elem *elem,
+ u16_t company, u16_t id);
+struct bt_mesh_model *bt_mesh_model_find(struct bt_mesh_elem *elem,
+ u16_t id);
+
+u16_t *bt_mesh_model_find_group(struct bt_mesh_model *mod, u16_t addr);
+
+bool bt_mesh_fixed_group_match(u16_t addr);
+
+void bt_mesh_model_foreach(void (*func)(struct bt_mesh_model *mod,
+ struct bt_mesh_elem *elem,
+ bool vnd, bool primary,
+ void *user_data),
+ void *user_data);
+
+s32_t bt_mesh_model_pub_period_get(struct bt_mesh_model *mod);
+
+void bt_mesh_comp_provision(u16_t addr);
+void bt_mesh_comp_unprovision(void);
+
+u16_t bt_mesh_primary_addr(void);
+
+struct bt_mesh_model *bt_mesh_model_get(bool vnd, u8_t elem_idx, u8_t mod_idx);
+
+const struct bt_mesh_comp *bt_mesh_comp_get(void);
+
+void bt_mesh_model_recv(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf);
+
+int bt_mesh_comp_register(const struct bt_mesh_comp *comp);
+#endif /* #if CONFIG_BT_MESH */
+#endif /* #ifndef _ACCESS_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/adv.c b/components/bt/ble_mesh/mesh_core/adv.c
new file mode 100644
index 0000000000..ee4725f2bb
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/adv.c
@@ -0,0 +1,431 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "freertos/xtensa_api.h"
+#include "freertos/FreeRTOSConfig.h"
+#include "freertos/FreeRTOS.h"
+#include "freertos/queue.h"
+#include "freertos/task.h"
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+
+#include "osi/thread.h"
+
+#include "mesh_util.h"
+
+#include "mesh_buf.h"
+#include "mesh.h"
+#include "mesh_bearer_adapt.h"
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_ADV)
+#include "mesh_trace.h"
+
+#include "adv.h"
+#include "foundation.h"
+#include "net.h"
+#include "beacon.h"
+#include "prov.h"
+#include "proxy.h"
+#include "mesh_hci.h"
+
+#include "sdkconfig.h"
+#include "provisioner_prov.h"
+#include "provisioner_proxy.h"
+#include "provisioner_beacon.h"
+
+#define BT_DATA_MESH_PROV 0x29 /* Mesh Provisioning PDU */
+#define BT_DATA_MESH_MESSAGE 0x2a /* Mesh Networking PDU */
+#define BT_DATA_MESH_BEACON 0x2b /* Mesh Beacon */
+
+
+/* Window and Interval are equal for continuous scanning */
+#define MESH_SCAN_INTERVAL 0x20//0x10 /*Change the scan interval to 20ms here just to reduce the packet loss rate */
+#define MESH_SCAN_WINDOW 0x20//0x10 /*Change the scan window to 20ms here just to reduce the packet loss rate */
+
+/* Convert from ms to 0.625ms units */
+#define ADV_INT(_ms) ((_ms) * 8 / 5)
+
+/* Pre-5.0 controllers enforce a minimum interval of 100ms
+ * whereas 5.0+ controllers can go down to 20ms.
+ */
+#define ADV_INT_DEFAULT K_MSEC(100)
+#define ADV_INT_FAST K_MSEC(20)
+
+/* TinyCrypt PRNG consumes a lot of stack space, so we need to have
+ * an increased call stack whenever it's used.
+ */
+#if defined(CONFIG_BT_HOST_CRYPTO)
+#define ADV_STACK_SIZE 768
+#else
+#define ADV_STACK_SIZE 512
+#endif
+
+static xQueueHandle xBleMeshQueue = 0;
+static const bt_addr_le_t *dev_addr;
+
+static const u8_t adv_type[] = {
+ [BT_MESH_ADV_PROV] = BT_DATA_MESH_PROV,
+ [BT_MESH_ADV_DATA] = BT_DATA_MESH_MESSAGE,
+ [BT_MESH_ADV_BEACON] = BT_DATA_MESH_BEACON,
+};
+
+NET_BUF_POOL_DEFINE(adv_buf_pool, CONFIG_BT_MESH_ADV_BUF_COUNT + 3 * CONFIG_BT_MESH_PBA_SAME_TIME,
+ BT_MESH_ADV_DATA_SIZE, BT_MESH_ADV_USER_DATA_SIZE, NULL);
+
+static struct bt_mesh_adv adv_pool[CONFIG_BT_MESH_ADV_BUF_COUNT + 3 * CONFIG_BT_MESH_PBA_SAME_TIME];
+
+static struct bt_mesh_adv *adv_alloc(int id)
+{
+ return &adv_pool[id];
+}
+
+static inline void adv_send_start(u16_t duration, int err,
+ const struct bt_mesh_send_cb *cb,
+ void *cb_data)
+{
+ if (cb && cb->start) {
+ cb->start(duration, err, cb_data);
+ }
+}
+
+static inline void adv_send_end(int err, const struct bt_mesh_send_cb *cb,
+ void *cb_data)
+{
+ if (cb && cb->end) {
+ cb->end(err, cb_data);
+ }
+}
+
+static inline int adv_send(struct net_buf *buf)
+{
+ const s32_t adv_int_min = ((bt_dev.hci_version >= BT_HCI_VERSION_5_0) ?
+ ADV_INT_FAST : ADV_INT_DEFAULT);
+ const struct bt_mesh_send_cb *cb = BT_MESH_ADV(buf)->cb;
+ void *cb_data = BT_MESH_ADV(buf)->cb_data;
+ struct bt_le_adv_param param;
+ u16_t duration, adv_int;
+ struct bt_data ad;
+ int err;
+
+ adv_int = max(adv_int_min, BT_MESH_ADV(buf)->adv_int);
+ duration = (BT_MESH_ADV(buf)->count + 1) * (adv_int + 10);
+
+ BT_DBG("type %u len %u: %s", BT_MESH_ADV(buf)->type,
+ buf->len, bt_hex(buf->data, buf->len));
+ BT_DBG("count %u interval %ums duration %ums",
+ BT_MESH_ADV(buf)->count + 1, adv_int, duration);
+
+ ad.type = adv_type[BT_MESH_ADV(buf)->type];
+ ad.data_len = buf->len;
+ ad.data = buf->data;
+
+ param.options = 0;
+ param.interval_min = ADV_INT(adv_int);
+ param.interval_max = param.interval_min;
+ param.own_addr = NULL;
+
+ err = bt_le_adv_start(¶m, &ad, 1, NULL, 0);
+ net_buf_unref(buf);
+ adv_send_start(duration, err, cb, cb_data);
+ if (err) {
+ BT_ERR("Advertising failed: err %d", err);
+ return err;
+ }
+
+ BT_DBG("Advertising started. Sleeping %u ms", duration);
+
+ k_sleep(duration);
+
+ err = bt_le_adv_stop();
+ adv_send_end(err, cb, cb_data);
+ if (err) {
+ BT_ERR("Stopping advertising failed: err %d", err);
+ /* If start adv successfully but stop failed, we think the data has been sent successfully */
+ return 0;
+ }
+
+ BT_DBG("Advertising stopped");
+ return 0;
+}
+
+/* Change by Espressif. The implementation of this function needs to be modified to
+ * FreeRTOS task implementation to use on ESP-IDF */
+static void adv_thread(void *p)
+{
+ BT_DBG("started");
+ int status;
+ struct net_buf **buf;
+ ble_mesh_msg_t msg = {0};
+ buf = (struct net_buf **)(&msg.arg);
+
+ while (1) {
+ *buf = NULL;
+#if CONFIG_BT_MESH_NODE
+ if (IS_ENABLED(CONFIG_BT_MESH_PROXY)) {
+ xQueueReceive(xBleMeshQueue, &msg, K_NO_WAIT);
+ while (!(*buf)) {
+ s32_t timeout;
+ BT_DBG("========");
+ timeout = bt_mesh_proxy_adv_start();
+ BT_DBG("=========Proxy Advertising up to %d ms", timeout);
+ xQueueReceive(xBleMeshQueue, &msg, timeout);
+ bt_mesh_proxy_adv_stop();
+ }
+ } else {
+ xQueueReceive(xBleMeshQueue, &msg, (portTickType)portMAX_DELAY);
+ }
+#else
+ xQueueReceive(xBleMeshQueue, &msg, (portTickType)portMAX_DELAY);
+#endif
+
+ if (!(*buf)) {
+ continue;
+ }
+
+ /* busy == 0 means this was canceled */
+ if (BT_MESH_ADV(*buf)->busy) {
+ BT_MESH_ADV(*buf)->busy = 0;
+ /*TODO: Currently we check status of adv_send, which has changed the original
+ * code of Zephyr, we need to find a better way in the future
+ * */
+ status = adv_send(*buf);
+ if (status) {
+ status = xQueueSendToFront(xBleMeshQueue, &msg, K_NO_WAIT);
+ if (status) {
+ BT_ERR("adv_send, xQueueSendToFront failed, status=0x%x", status);
+ }
+ }
+ }
+
+ /* Give other threads a chance to run */
+ taskYIELD();
+ }
+}
+
+void bt_mesh_adv_update(void)
+{
+ BT_DBG("%s", __func__);
+ ble_mesh_msg_t msg = {0};
+ msg.arg = NULL;
+ // Change by Espressif, should used the QueueSend in the ESP-IDF architecture.
+ ble_mesh_task_post(&msg, 0);
+}
+
+struct net_buf *bt_mesh_adv_create_from_pool(struct net_buf_pool *pool,
+ bt_mesh_adv_alloc_t get_id,
+ enum bt_mesh_adv_type type,
+ u8_t xmit_count, u8_t xmit_int,
+ s32_t timeout)
+{
+ struct bt_mesh_adv *adv;
+ struct net_buf *buf;
+
+ buf = net_buf_alloc(pool, timeout);
+ if (!buf) {
+ return NULL;
+ }
+
+ BT_DBG("%s, pool_id = %p, buf_count = %d, uinit_count = %d", __func__,
+ buf->pool_id, pool->buf_count, pool->uninit_count);
+ // adv = get_id(pool->buf_count - pool->uninit_count);
+ /* Change by Espressif. Use buf->index to get corresponding adv_pool */
+ adv = get_id(buf->index);
+ BT_MESH_ADV(buf) = adv;
+
+ memset(adv, 0, sizeof(*adv));
+
+ adv->type = type;
+ adv->count = xmit_count;
+ adv->adv_int = xmit_int;
+
+ return buf;
+}
+
+struct net_buf *bt_mesh_adv_create(enum bt_mesh_adv_type type, u8_t xmit_count,
+ u8_t xmit_int, s32_t timeout)
+{
+ return bt_mesh_adv_create_from_pool(&adv_buf_pool, adv_alloc, type,
+ xmit_count, xmit_int, timeout);
+}
+
+void ble_mesh_task_post(ble_mesh_msg_t *msg, uint32_t timeout)
+{
+ BT_DBG("%s", __func__);
+ if (xQueueSend(xBleMeshQueue, msg, timeout) != pdTRUE) {
+ BT_ERR("Ble Mesh Post failed\n");
+ return;
+ }
+}
+
+void bt_mesh_adv_send(struct net_buf *buf, const struct bt_mesh_send_cb *cb,
+ void *cb_data)
+{
+ ble_mesh_msg_t msg = {0};
+ struct net_buf *send_buf = NULL;
+ BT_DBG("type 0x%02x len %u: %s", BT_MESH_ADV(buf)->type, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ BT_MESH_ADV(buf)->cb = cb;
+ BT_MESH_ADV(buf)->cb_data = cb_data;
+ BT_MESH_ADV(buf)->busy = 1;
+ send_buf = net_buf_ref(buf);
+ msg.arg = (void *)send_buf;
+ //net_buf_put(NULL, net_buf_ref(buf));
+ /* Change by Espressif. The ESP-IDF should used the QueueSend to sent the msg. */
+ ble_mesh_task_post(&msg, portMAX_DELAY);
+ /* Deleted the net_buf_put function, we don't used it in the ESP-IDF architecture */
+ //net_buf_put(NULL, net_buf_ref(buf));
+}
+
+const bt_addr_le_t *bt_mesh_pba_get_addr(void)
+{
+ return dev_addr;
+}
+
+static void bt_mesh_scan_cb(const bt_addr_le_t *addr, s8_t rssi,
+ u8_t adv_type, struct net_buf_simple *buf)
+{
+#if CONFIG_BT_MESH_PROVISIONER && CONFIG_BT_MESH_PB_GATT
+ u16_t uuid = 0;
+#endif
+
+ if (adv_type != BT_LE_ADV_NONCONN_IND && adv_type != BT_LE_ADV_IND) {
+ return;
+ }
+
+ BT_DBG("%s, len %u: %s", __func__, buf->len, bt_hex(buf->data, buf->len));
+
+ dev_addr = addr;
+
+ while (buf->len > 1) {
+ struct net_buf_simple_state state;
+ u8_t len, type;
+
+ len = net_buf_simple_pull_u8(buf);
+ /* Check for early termination */
+ if (len == 0) {
+ return;
+ }
+
+ if (len > buf->len || buf->len < 1) {
+ BT_WARN("AD malformed");
+ return;
+ }
+
+ net_buf_simple_save(buf, &state);
+
+ type = net_buf_simple_pull_u8(buf);
+ BT_DBG("%s, type = %x", __func__, type);
+ buf->len = len - 1;
+
+#if 0
+ /* TODO: Check with BLE Mesh BQB test cases */
+ if ((type == BT_DATA_MESH_PROV || type == BT_DATA_MESH_MESSAGE ||
+ type == BT_DATA_MESH_BEACON) && (adv_type != BT_LE_ADV_NONCONN_IND)) {
+ BT_DBG("%s, ignore BLE Mesh packet (type 0x%02x) with adv_type 0x%02x",
+ __func__, type, adv_type);
+ return;
+ }
+#endif
+
+ switch (type) {
+ case BT_DATA_MESH_MESSAGE:
+ bt_mesh_net_recv(buf, rssi, BT_MESH_NET_IF_ADV);
+ break;
+#if CONFIG_BT_MESH_PB_ADV
+ case BT_DATA_MESH_PROV:
+#if CONFIG_BT_MESH_NODE
+ if (!bt_mesh_is_provisioner_en()) {
+ bt_mesh_pb_adv_recv(buf);
+ }
+#endif
+#if CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ provisioner_pb_adv_recv(buf);
+ }
+#endif
+ break;
+#endif /* CONFIG_BT_MESH_PB_ADV */
+ case BT_DATA_MESH_BEACON:
+#if CONFIG_BT_MESH_NODE
+ if (!bt_mesh_is_provisioner_en()) {
+ bt_mesh_beacon_recv(buf);
+ }
+#endif
+#if CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ provisioner_beacon_recv(buf);
+ }
+#endif
+ break;
+#if CONFIG_BT_MESH_PROVISIONER && CONFIG_BT_MESH_PB_GATT
+ case BT_DATA_FLAGS:
+ if (bt_mesh_is_provisioner_en()) {
+ if (!provisioner_flags_match(buf)) {
+ BT_DBG("Flags mismatch, ignore this adv pkt");
+ return;
+ }
+ }
+ break;
+ case BT_DATA_SERVICE_UUID:
+ if (bt_mesh_is_provisioner_en()) {
+ uuid = provisioner_srv_uuid_recv(buf);
+ if (!uuid) {
+ BT_DBG("Service UUID mismatch, ignore this adv pkt");
+ return;
+ }
+ }
+ break;
+ case BT_DATA_SERVICE_DATA:
+ if (bt_mesh_is_provisioner_en()) {
+ provisioner_srv_data_recv(buf, addr, uuid);
+ }
+ break;
+#endif /* CONFIG_BT_MESH_PROVISIONER && CONFIG_BT_MESH_PB_GATT */
+ default:
+ break;
+ }
+
+ net_buf_simple_restore(buf, &state);
+ net_buf_simple_pull(buf, len);
+ }
+
+ return;
+}
+
+void bt_mesh_adv_init(void)
+{
+ /* Change by Espressif, we should used the FreeRTOS create task method to use task */
+ xBleMeshQueue = xQueueCreate(150, sizeof(ble_mesh_msg_t));
+ xTaskCreatePinnedToCore(adv_thread, "BLE_Mesh_Adv_task", 2048, NULL,
+ configMAX_PRIORITIES - 7, NULL, TASK_PINNED_TO_CORE);
+}
+
+int bt_mesh_scan_enable(void)
+{
+ struct bt_le_scan_param scan_param = {
+ .type = BT_HCI_LE_SCAN_PASSIVE,
+ .filter_dup = BT_HCI_LE_SCAN_FILTER_DUP_DISABLE,
+ .interval = MESH_SCAN_INTERVAL,
+ .window = MESH_SCAN_WINDOW
+ };
+
+ BT_DBG("%s", __func__);
+
+ return bt_le_scan_start(&scan_param, bt_mesh_scan_cb);
+}
+
+int bt_mesh_scan_disable(void)
+{
+ BT_DBG("%s", __func__);
+
+ return bt_le_scan_stop();
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/adv.h b/components/bt/ble_mesh/mesh_core/adv.h
new file mode 100644
index 0000000000..cd225cbd0d
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/adv.h
@@ -0,0 +1,82 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "mesh_bearer_adapt.h"
+
+/* Maximum advertising data payload for a single data type */
+#define BT_MESH_ADV_DATA_SIZE 29
+
+/* The user data is a pointer (4 bytes) to struct bt_mesh_adv */
+#define BT_MESH_ADV_USER_DATA_SIZE 4
+
+#define BT_MESH_ADV(buf) (*(struct bt_mesh_adv **)net_buf_user_data(buf))
+
+typedef struct ble_mesh_msg {
+ uint8_t sig; //event signal
+ uint8_t aid; //application id
+ uint8_t pid; //profile id
+ uint8_t act; //profile action, defined in seprerate header files
+ void *arg; //param for btc function or function param
+} ble_mesh_msg_t;
+
+enum bt_mesh_adv_type {
+ BT_MESH_ADV_PROV,
+ BT_MESH_ADV_DATA,
+ BT_MESH_ADV_BEACON,
+};
+
+typedef void (*bt_mesh_adv_func_t)(struct net_buf *buf, u16_t duration,
+ int err, void *user_data);
+
+struct bt_mesh_adv {
+ const struct bt_mesh_send_cb *cb;
+ void *cb_data;
+
+ u8_t type: 2,
+ busy: 1;
+ u8_t count: 3,
+ adv_int: 5;
+ union {
+ /* Address, used e.g. for Friend Queue messages */
+ u16_t addr;
+
+ /* For transport layer segment sending */
+ struct {
+ u8_t attempts;
+ } seg;
+ };
+};
+
+typedef struct bt_mesh_adv *(*bt_mesh_adv_alloc_t)(int id);
+
+/* xmit_count: Number of retransmissions, i.e. 0 == 1 transmission */
+struct net_buf *bt_mesh_adv_create(enum bt_mesh_adv_type type, u8_t xmit_count,
+ u8_t xmit_int, s32_t timeout);
+
+struct net_buf *bt_mesh_adv_create_from_pool(struct net_buf_pool *pool,
+ bt_mesh_adv_alloc_t get_id,
+ enum bt_mesh_adv_type type,
+ u8_t xmit_count, u8_t xmit_int,
+ s32_t timeout);
+
+void bt_mesh_adv_send(struct net_buf *buf, const struct bt_mesh_send_cb *cb,
+ void *cb_data);
+
+const bt_addr_le_t *bt_mesh_pba_get_addr(void);
+
+void bt_mesh_adv_update(void);
+
+void bt_mesh_adv_init(void);
+
+int bt_mesh_scan_enable(void);
+
+int bt_mesh_scan_disable(void);
+
+void ble_mesh_task_post(ble_mesh_msg_t *msg, uint32_t timeout);
+
diff --git a/components/bt/ble_mesh/mesh_core/base64.c b/components/bt/ble_mesh/mesh_core/base64.c
new file mode 100644
index 0000000000..83748f56b2
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/base64.c
@@ -0,0 +1,217 @@
+/*
+ * RFC 1521 base64 encoding/decoding
+ *
+ * Copyright (C) 2018, Nordic Semiconductor ASA
+ * Copyright (C) 2006-2015, ARM Limited, All Rights Reserved
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may
+ * not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ *
+ * Adapted for Zephyr by Carles Cufi (carles.cufi@nordicsemi.no)
+ * - Removed mbedtls_ prefixes
+ * - Reworked coding style
+ */
+
+#include
+#include
+#include
+
+static const u8_t base64_enc_map[64] = {
+ 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',
+ 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T',
+ 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd',
+ 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n',
+ 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x',
+ 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7',
+ '8', '9', '+', '/'
+};
+
+static const u8_t base64_dec_map[128] = {
+ 127, 127, 127, 127, 127, 127, 127, 127, 127, 127,
+ 127, 127, 127, 127, 127, 127, 127, 127, 127, 127,
+ 127, 127, 127, 127, 127, 127, 127, 127, 127, 127,
+ 127, 127, 127, 127, 127, 127, 127, 127, 127, 127,
+ 127, 127, 127, 62, 127, 127, 127, 63, 52, 53,
+ 54, 55, 56, 57, 58, 59, 60, 61, 127, 127,
+ 127, 64, 127, 127, 127, 0, 1, 2, 3, 4,
+ 5, 6, 7, 8, 9, 10, 11, 12, 13, 14,
+ 15, 16, 17, 18, 19, 20, 21, 22, 23, 24,
+ 25, 127, 127, 127, 127, 127, 127, 26, 27, 28,
+ 29, 30, 31, 32, 33, 34, 35, 36, 37, 38,
+ 39, 40, 41, 42, 43, 44, 45, 46, 47, 48,
+ 49, 50, 51, 127, 127, 127, 127, 127
+};
+
+#define BASE64_SIZE_T_MAX ((size_t) -1) /* SIZE_T_MAX is not standard */
+
+/*
+ * Encode a buffer into base64 format
+ */
+int base64_encode(u8_t *dst, size_t dlen, size_t *olen, const u8_t *src,
+ size_t slen)
+{
+ size_t i, n;
+ int C1, C2, C3;
+ u8_t *p;
+
+ if (slen == 0) {
+ *olen = 0;
+ return 0;
+ }
+
+ n = slen / 3 + (slen % 3 != 0);
+
+ if (n > (BASE64_SIZE_T_MAX - 1) / 4) {
+ *olen = BASE64_SIZE_T_MAX;
+ return -ENOMEM;
+ }
+
+ n *= 4;
+
+ if ((dlen < n + 1) || (!dst)) {
+ *olen = n + 1;
+ return -ENOMEM;
+ }
+
+ n = (slen / 3) * 3;
+
+ for (i = 0, p = dst; i < n; i += 3) {
+ C1 = *src++;
+ C2 = *src++;
+ C3 = *src++;
+
+ *p++ = base64_enc_map[(C1 >> 2) & 0x3F];
+ *p++ = base64_enc_map[(((C1 & 3) << 4) + (C2 >> 4)) & 0x3F];
+ *p++ = base64_enc_map[(((C2 & 15) << 2) + (C3 >> 6)) & 0x3F];
+ *p++ = base64_enc_map[C3 & 0x3F];
+ }
+
+ if (i < slen) {
+ C1 = *src++;
+ C2 = ((i + 1) < slen) ? *src++ : 0;
+
+ *p++ = base64_enc_map[(C1 >> 2) & 0x3F];
+ *p++ = base64_enc_map[(((C1 & 3) << 4) + (C2 >> 4)) & 0x3F];
+
+ if ((i + 1) < slen) {
+ *p++ = base64_enc_map[((C2 & 15) << 2) & 0x3F];
+ } else {
+ *p++ = '=';
+ }
+
+ *p++ = '=';
+ }
+
+ *olen = p - dst;
+ *p = 0;
+
+ return 0;
+}
+
+/*
+ * Decode a base64-formatted buffer
+ */
+int base64_decode(u8_t *dst, size_t dlen, size_t *olen, const u8_t *src,
+ size_t slen)
+{
+ size_t i, n;
+ u32_t j, x;
+ u8_t *p;
+
+ /* First pass: check for validity and get output length */
+ for (i = n = j = 0; i < slen; i++) {
+ /* Skip spaces before checking for EOL */
+ x = 0;
+ while (i < slen && src[i] == ' ') {
+ ++i;
+ ++x;
+ }
+
+ /* Spaces at end of buffer are OK */
+ if (i == slen) {
+ break;
+ }
+
+ if ((slen - i) >= 2 && src[i] == '\r' && src[i + 1] == '\n') {
+ continue;
+ }
+
+ if (src[i] == '\n') {
+ continue;
+ }
+
+ /* Space inside a line is an error */
+ if (x != 0) {
+ return -EINVAL;
+ }
+
+ if (src[i] == '=' && ++j > 2) {
+ return -EINVAL;
+ }
+
+ if (src[i] > 127 || base64_dec_map[src[i]] == 127) {
+ return -EINVAL;
+ }
+
+ if (base64_dec_map[src[i]] < 64 && j != 0) {
+ return -EINVAL;
+ }
+
+ n++;
+ }
+
+ if (n == 0) {
+ *olen = 0;
+ return 0;
+ }
+
+ /* The following expression is to calculate the following formula
+ * without risk of integer overflow in n:
+ * n = ( ( n * 6 ) + 7 ) >> 3;
+ */
+ n = (6 * (n >> 3)) + ((6 * (n & 0x7) + 7) >> 3);
+ n -= j;
+
+ if (dst == NULL || dlen < n) {
+ *olen = n;
+ return -ENOMEM;
+ }
+
+ for (j = 3, n = x = 0, p = dst; i > 0; i--, src++) {
+
+ if (*src == '\r' || *src == '\n' || *src == ' ') {
+ continue;
+ }
+
+ j -= (base64_dec_map[*src] == 64);
+ x = (x << 6) | (base64_dec_map[*src] & 0x3F);
+
+ if (++n == 4) {
+ n = 0;
+ if (j > 0) {
+ *p++ = (unsigned char)(x >> 16);
+ }
+ if (j > 1) {
+ *p++ = (unsigned char)(x >> 8);
+ }
+ if (j > 2) {
+ *p++ = (unsigned char)(x);
+ }
+ }
+ }
+
+ *olen = p - dst;
+
+ return 0;
+}
+
diff --git a/components/bt/ble_mesh/mesh_core/beacon.c b/components/bt/ble_mesh/mesh_core/beacon.c
new file mode 100644
index 0000000000..68ec03e350
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/beacon.c
@@ -0,0 +1,402 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+
+#include "mesh_util.h"
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#include "mesh_main.h"
+
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_BEACON)
+#include "mesh_trace.h"
+
+#include "adv.h"
+#include "mesh.h"
+#include "net.h"
+#include "prov.h"
+#include "crypto.h"
+#include "beacon.h"
+#include "foundation.h"
+
+#include "provisioner_prov.h"
+
+#if CONFIG_BT_MESH_NODE
+
+#if defined(CONFIG_BT_MESH_FAST_PROV)
+#define UNPROVISIONED_INTERVAL K_SECONDS(3)
+#else
+#define UNPROVISIONED_INTERVAL K_SECONDS(5)
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+#define PROVISIONED_INTERVAL K_SECONDS(10)
+
+#define BEACON_TYPE_UNPROVISIONED 0x00
+#define BEACON_TYPE_SECURE 0x01
+
+/* 3 transmissions, 20ms interval */
+#define UNPROV_XMIT_COUNT 2
+#define UNPROV_XMIT_INT 20
+
+/* 1 transmission, 20ms interval */
+#define PROV_XMIT_COUNT 0
+#define PROV_XMIT_INT 20
+
+static struct k_delayed_work beacon_timer;
+
+static struct bt_mesh_subnet *cache_check(u8_t data[21])
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ if (!memcmp(sub->beacon_cache, data, 21)) {
+ return sub;
+ }
+ }
+
+ return NULL;
+}
+
+static void cache_add(u8_t data[21], struct bt_mesh_subnet *sub)
+{
+ memcpy(sub->beacon_cache, data, 21);
+}
+
+static void beacon_complete(int err, void *user_data)
+{
+ struct bt_mesh_subnet *sub = user_data;
+
+ BT_DBG("err %d", err);
+
+ sub->beacon_sent = k_uptime_get_32();
+}
+
+void bt_mesh_beacon_create(struct bt_mesh_subnet *sub,
+ struct net_buf_simple *buf)
+{
+ u8_t flags = bt_mesh_net_flags(sub);
+ struct bt_mesh_subnet_keys *keys;
+
+ net_buf_simple_add_u8(buf, BEACON_TYPE_SECURE);
+
+ if (sub->kr_flag) {
+ keys = &sub->keys[1];
+ } else {
+ keys = &sub->keys[0];
+ }
+
+ net_buf_simple_add_u8(buf, flags);
+
+ /* Network ID */
+ net_buf_simple_add_mem(buf, keys->net_id, 8);
+
+ /* IV Index */
+ net_buf_simple_add_be32(buf, bt_mesh.iv_index);
+
+ net_buf_simple_add_mem(buf, sub->auth, 8);
+
+ BT_DBG("net_idx 0x%04x flags 0x%02x NetID %s", sub->net_idx,
+ flags, bt_hex(keys->net_id, 8));
+ BT_DBG("IV Index 0x%08x Auth %s", bt_mesh.iv_index,
+ bt_hex(sub->auth, 8));
+}
+
+/* If the interval has passed or is within 5 seconds from now send a beacon */
+#define BEACON_THRESHOLD(sub) (K_SECONDS(10 * ((sub)->beacons_last + 1)) - K_SECONDS(5))
+
+static int secure_beacon_send(void)
+{
+ static const struct bt_mesh_send_cb send_cb = {
+ .end = beacon_complete,
+ };
+ u32_t now = k_uptime_get_32();
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+ struct net_buf *buf;
+ u32_t time_diff;
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ time_diff = now - sub->beacon_sent;
+ if (time_diff < K_SECONDS(600) &&
+ time_diff < BEACON_THRESHOLD(sub)) {
+ continue;
+ }
+
+ buf = bt_mesh_adv_create(BT_MESH_ADV_BEACON, PROV_XMIT_COUNT,
+ PROV_XMIT_INT, K_NO_WAIT);
+ if (!buf) {
+ BT_ERR("Unable to allocate beacon buffer");
+ return -ENOBUFS;
+ }
+
+ bt_mesh_beacon_create(sub, &buf->b);
+
+ bt_mesh_adv_send(buf, &send_cb, sub);
+ net_buf_unref(buf);
+ }
+
+ return 0;
+}
+
+static int unprovisioned_beacon_send(void)
+{
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ struct net_buf *buf;
+ struct bt_mesh_adv *adv;
+ BT_DBG("%s", __func__);
+
+ buf = bt_mesh_adv_create(BT_MESH_ADV_BEACON, UNPROV_XMIT_COUNT,
+ UNPROV_XMIT_INT, K_NO_WAIT);
+ if (!buf) {
+ BT_ERR("Unable to allocate beacon buffer");
+ return -ENOBUFS;
+ }
+ // Change by Espressif, added the following code for the debug used.
+ adv = BT_MESH_ADV(buf);
+ BT_DBG("%s, adv->type = %d, adv->count = %d, adv->adv_int = %d", __func__,
+ adv->type, adv->count, adv->adv_int);
+ net_buf_add_u8(buf, BEACON_TYPE_UNPROVISIONED);
+ net_buf_add_mem(buf, bt_mesh_prov_get_uuid(), 16);
+
+ /* OOB Info (2 bytes) + URI Hash (4 bytes) */
+ memset(net_buf_add(buf, 2 + 4), 0, 2 + 4);
+
+ bt_mesh_adv_send(buf, NULL, NULL);
+ net_buf_unref(buf);
+
+#endif /* CONFIG_BT_MESH_PB_ADV */
+ return 0;
+}
+
+static void update_beacon_observation(void)
+{
+ static bool first_half;
+ int i;
+
+ /* Observation period is 20 seconds, whereas the beacon timer
+ * runs every 10 seconds. We process what's happened during the
+ * window only after the second half.
+ */
+ first_half = !first_half;
+ if (first_half) {
+ return;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ sub->beacons_last = sub->beacons_cur;
+ sub->beacons_cur = 0;
+ }
+}
+
+static void beacon_send(struct k_work *work)
+{
+ /* Don't send anything if there is an active provisioning link */
+ if (IS_ENABLED(CONFIG_BT_MESH_PROV) && bt_prov_active()) {
+ k_delayed_work_submit(&beacon_timer, UNPROVISIONED_INTERVAL);
+ return;
+ }
+
+ BT_DBG("%s", __func__);
+
+ if (bt_mesh_is_provisioned()) {
+ update_beacon_observation();
+ secure_beacon_send();
+
+ /* Only resubmit if beaconing is still enabled */
+ if (bt_mesh_beacon_get() == BT_MESH_BEACON_ENABLED ||
+ bt_mesh.ivu_initiator) {
+ k_delayed_work_submit(&beacon_timer,
+ PROVISIONED_INTERVAL);
+ }
+ } else {
+ unprovisioned_beacon_send();
+ k_delayed_work_submit(&beacon_timer, UNPROVISIONED_INTERVAL);
+ }
+
+}
+
+static void secure_beacon_recv(struct net_buf_simple *buf)
+{
+ u8_t *data, *net_id, *auth;
+ struct bt_mesh_subnet *sub;
+ u32_t iv_index;
+ bool new_key, kr_change, iv_change;
+ u8_t flags;
+
+ if (buf->len < 21) {
+ BT_ERR("Too short secure beacon (len %u)", buf->len);
+ return;
+ }
+
+ sub = cache_check(buf->data);
+ if (sub) {
+ /* We've seen this beacon before - just update the stats */
+ goto update_stats;
+ }
+
+ /* So we can add to the cache if auth matches */
+ data = buf->data;
+
+ flags = net_buf_simple_pull_u8(buf);
+ net_id = buf->data;
+ net_buf_simple_pull(buf, 8);
+ iv_index = net_buf_simple_pull_be32(buf);
+ auth = buf->data;
+
+ BT_DBG("flags 0x%02x id %s iv_index 0x%08x",
+ flags, bt_hex(net_id, 8), iv_index);
+
+ sub = bt_mesh_subnet_find(net_id, flags, iv_index, auth, &new_key);
+ if (!sub) {
+ BT_DBG("No subnet that matched beacon");
+ return;
+ }
+
+ if (sub->kr_phase == BT_MESH_KR_PHASE_2 && !new_key) {
+ BT_WARN("Ignoring Phase 2 KR Update secured using old key");
+ return;
+ }
+
+ cache_add(data, sub);
+
+ /* If there is NetKey0, accept initiation only from it */
+ if (bt_mesh_subnet_get(BT_MESH_KEY_PRIMARY) &&
+ sub->net_idx != BT_MESH_KEY_PRIMARY) {
+ BT_WARN("Ignoring secure beacon on non-primary subnet");
+ goto update_stats;
+ }
+
+ BT_DBG("net_idx 0x%04x iv_index 0x%08x, current iv_index 0x%08x",
+ sub->net_idx, iv_index, bt_mesh.iv_index);
+
+ if (bt_mesh.ivu_initiator &&
+ bt_mesh.iv_update == BT_MESH_IV_UPDATE(flags)) {
+ bt_mesh_beacon_ivu_initiator(false);
+ }
+
+ iv_change = bt_mesh_net_iv_update(iv_index, BT_MESH_IV_UPDATE(flags));
+
+ kr_change = bt_mesh_kr_update(sub, BT_MESH_KEY_REFRESH(flags), new_key);
+ if (kr_change) {
+ bt_mesh_net_beacon_update(sub);
+ }
+
+ if (iv_change) {
+ /* Update all subnets */
+ bt_mesh_net_sec_update(NULL);
+ } else if (kr_change) {
+ /* Key Refresh without IV Update only impacts one subnet */
+ bt_mesh_net_sec_update(sub);
+ }
+
+update_stats:
+ if (bt_mesh_beacon_get() == BT_MESH_BEACON_ENABLED &&
+ sub->beacons_cur < 0xff) {
+ sub->beacons_cur++;
+ }
+}
+
+void bt_mesh_beacon_recv(struct net_buf_simple *buf)
+{
+ u8_t type;
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (buf->len < 1) {
+ BT_ERR("Too short beacon");
+ return;
+ }
+
+ type = net_buf_simple_pull_u8(buf);
+ switch (type) {
+ case BEACON_TYPE_UNPROVISIONED:
+ BT_DBG("Ignoring unprovisioned device beacon");
+ break;
+ case BEACON_TYPE_SECURE:
+ secure_beacon_recv(buf);
+ break;
+ default:
+ BT_WARN("Unknown beacon type 0x%02x", type);
+ break;
+ }
+}
+
+void bt_mesh_beacon_init(void)
+{
+ k_delayed_work_init(&beacon_timer, beacon_send);
+}
+
+void bt_mesh_beacon_ivu_initiator(bool enable)
+{
+ bt_mesh.ivu_initiator = enable;
+
+ if (enable) {
+ k_work_submit(&beacon_timer.work);
+ } else if (bt_mesh_beacon_get() == BT_MESH_BEACON_DISABLED) {
+ k_delayed_work_cancel(&beacon_timer);
+ }
+}
+
+void bt_mesh_beacon_enable(void)
+{
+ int i;
+
+ if (!bt_mesh_is_provisioned()) {
+ k_work_submit(&beacon_timer.work);
+ return;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ sub->beacons_last = 0;
+ sub->beacons_cur = 0;
+
+ bt_mesh_net_beacon_update(sub);
+ }
+
+ k_work_submit(&beacon_timer.work);
+}
+
+void bt_mesh_beacon_disable(void)
+{
+ if (!bt_mesh.ivu_initiator) {
+ k_delayed_work_cancel(&beacon_timer);
+ }
+}
+
+#endif /* CONFIG_BT_MESH_NODE */
+
+#endif /* #if CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/beacon.h b/components/bt/ble_mesh/mesh_core/beacon.h
new file mode 100644
index 0000000000..76b8b49d65
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/beacon.h
@@ -0,0 +1,19 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+void bt_mesh_beacon_enable(void);
+void bt_mesh_beacon_disable(void);
+
+void bt_mesh_beacon_ivu_initiator(bool enable);
+
+void bt_mesh_beacon_recv(struct net_buf_simple *buf);
+
+void bt_mesh_beacon_create(struct bt_mesh_subnet *sub,
+ struct net_buf_simple *buf);
+
+void bt_mesh_beacon_init(void);
diff --git a/components/bt/ble_mesh/mesh_core/cfg_cli.c b/components/bt/ble_mesh/mesh_core/cfg_cli.c
new file mode 100644
index 0000000000..0387cc101f
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/cfg_cli.c
@@ -0,0 +1,1665 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+#include "mesh_types.h"
+#include "mesh_util.h"
+#include "mesh_kernel.h"
+
+#include "mesh.h"
+#include "sdkconfig.h"
+#include "osi/allocator.h"
+
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_MODEL)
+#include "mesh_trace.h"
+#include "cfg_cli.h"
+#include "foundation.h"
+#include "common.h"
+#include "btc_ble_mesh_config_client.h"
+
+#define CID_NVAL 0xffff
+
+s32_t config_msg_timeout;
+
+static bt_mesh_config_client_t *cli;
+
+static const bt_mesh_client_op_pair_t cfg_op_pair[] = {
+ { OP_BEACON_GET, OP_BEACON_STATUS },
+ { OP_BEACON_SET, OP_BEACON_STATUS },
+ { OP_DEV_COMP_DATA_GET, OP_DEV_COMP_DATA_STATUS },
+ { OP_DEFAULT_TTL_GET, OP_DEFAULT_TTL_STATUS },
+ { OP_DEFAULT_TTL_SET, OP_DEFAULT_TTL_STATUS },
+ { OP_GATT_PROXY_GET, OP_GATT_PROXY_STATUS },
+ { OP_GATT_PROXY_SET, OP_GATT_PROXY_STATUS },
+ { OP_RELAY_GET, OP_RELAY_STATUS },
+ { OP_RELAY_SET, OP_RELAY_STATUS },
+ { OP_MOD_PUB_GET, OP_MOD_PUB_STATUS },
+ { OP_MOD_PUB_SET, OP_MOD_PUB_STATUS },
+ { OP_MOD_PUB_VA_SET, OP_MOD_PUB_STATUS },
+ { OP_MOD_SUB_ADD, OP_MOD_SUB_STATUS },
+ { OP_MOD_SUB_VA_ADD, OP_MOD_SUB_STATUS },
+ { OP_MOD_SUB_DEL, OP_MOD_SUB_STATUS },
+ { OP_MOD_SUB_VA_DEL, OP_MOD_SUB_STATUS },
+ { OP_MOD_SUB_OVERWRITE, OP_MOD_SUB_STATUS },
+ { OP_MOD_SUB_VA_OVERWRITE, OP_MOD_SUB_STATUS },
+ { OP_MOD_SUB_DEL_ALL, OP_MOD_SUB_STATUS },
+ { OP_MOD_SUB_GET, OP_MOD_SUB_LIST },
+ { OP_MOD_SUB_GET_VND, OP_MOD_SUB_LIST_VND },
+ { OP_NET_KEY_ADD, OP_NET_KEY_STATUS },
+ { OP_NET_KEY_UPDATE, OP_NET_KEY_STATUS },
+ { OP_NET_KEY_DEL, OP_NET_KEY_STATUS },
+ { OP_NET_KEY_GET, OP_NET_KEY_LIST },
+ { OP_APP_KEY_ADD, OP_APP_KEY_STATUS },
+ { OP_APP_KEY_UPDATE, OP_APP_KEY_STATUS },
+ { OP_APP_KEY_DEL, OP_APP_KEY_STATUS },
+ { OP_APP_KEY_GET, OP_APP_KEY_LIST },
+ { OP_NODE_IDENTITY_GET, OP_NODE_IDENTITY_STATUS },
+ { OP_NODE_IDENTITY_SET, OP_NODE_IDENTITY_STATUS },
+ { OP_MOD_APP_BIND, OP_MOD_APP_STATUS },
+ { OP_MOD_APP_UNBIND, OP_MOD_APP_STATUS },
+ { OP_SIG_MOD_APP_GET, OP_SIG_MOD_APP_LIST },
+ { OP_VND_MOD_APP_GET, OP_VND_MOD_APP_LIST },
+ { OP_NODE_RESET, OP_NODE_RESET_STATUS },
+ { OP_FRIEND_GET, OP_FRIEND_STATUS },
+ { OP_FRIEND_SET, OP_FRIEND_STATUS },
+ { OP_KRP_GET, OP_KRP_STATUS },
+ { OP_KRP_SET, OP_KRP_STATUS },
+ { OP_HEARTBEAT_PUB_GET, OP_HEARTBEAT_PUB_STATUS },
+ { OP_HEARTBEAT_PUB_SET, OP_HEARTBEAT_PUB_STATUS },
+ { OP_HEARTBEAT_SUB_GET, OP_HEARTBEAT_SUB_STATUS },
+ { OP_HEARTBEAT_SUB_SET, OP_HEARTBEAT_SUB_STATUS },
+ { OP_LPN_TIMEOUT_GET, OP_LPN_TIMEOUT_STATUS },
+ { OP_NET_TRANSMIT_GET, OP_NET_TRANSMIT_STATUS },
+ { OP_NET_TRANSMIT_SET, OP_NET_TRANSMIT_STATUS },
+};
+
+static void timeout_handler(struct k_work *work)
+{
+ bt_mesh_config_client_t *client = NULL;
+ config_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+
+ BT_WARN("Receive configuration status message timeout");
+
+ node = CONTAINER_OF(work, bt_mesh_client_node_t, timer.work);
+ if (!node || !node->ctx.model) {
+ BT_ERR("%s: node parameter is NULL", __func__);
+ return;
+ }
+
+ client = (bt_mesh_config_client_t *)node->ctx.model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (config_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: internal_data is NULL", __func__);
+ return;
+ }
+
+ bt_mesh_callback_config_status_to_btc(node->opcode, 0x03, node->ctx.model,
+ &node->ctx, NULL, 0);
+
+ bt_mesh_client_free_node(&internal->queue, node);
+
+ return;
+}
+
+static void cfg_client_cancel(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ void *status, size_t len)
+{
+ config_internal_data_t *data = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ struct net_buf_simple buf = {0};
+ u8_t evt_type = 0xFF;
+
+ if (!model || !ctx) {
+ BT_ERR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ data = (config_internal_data_t *)cli->internal_data;
+ if (!data) {
+ BT_ERR("%s: config client internal_data is NULL", __func__);
+ return;
+ }
+
+ /* If it is a publish message, sent to the user directly. */
+ buf.data = (u8_t *)status;
+ buf.len = (u16_t)len;
+ node = bt_mesh_is_model_message_publish(model, ctx, &buf, true);
+ if (!node) {
+ BT_DBG("Unexpected config status message 0x%x", ctx->recv_op);
+ } else {
+ switch (node->opcode) {
+ case OP_BEACON_GET:
+ case OP_DEV_COMP_DATA_GET:
+ case OP_DEFAULT_TTL_GET:
+ case OP_GATT_PROXY_GET:
+ case OP_RELAY_GET:
+ case OP_MOD_PUB_GET:
+ case OP_MOD_SUB_GET:
+ case OP_MOD_SUB_GET_VND:
+ case OP_NET_KEY_GET:
+ case OP_APP_KEY_GET:
+ case OP_NODE_IDENTITY_GET:
+ case OP_SIG_MOD_APP_GET:
+ case OP_VND_MOD_APP_GET:
+ case OP_FRIEND_GET:
+ case OP_KRP_GET:
+ case OP_HEARTBEAT_PUB_GET:
+ case OP_HEARTBEAT_SUB_GET:
+ case OP_LPN_TIMEOUT_GET:
+ case OP_NET_TRANSMIT_GET:
+ evt_type = 0x00;
+ break;
+ case OP_BEACON_SET:
+ case OP_DEFAULT_TTL_SET:
+ case OP_GATT_PROXY_SET:
+ case OP_RELAY_SET:
+ case OP_MOD_PUB_SET:
+ case OP_MOD_PUB_VA_SET:
+ case OP_MOD_SUB_ADD:
+ case OP_MOD_SUB_VA_ADD:
+ case OP_MOD_SUB_DEL:
+ case OP_MOD_SUB_VA_DEL:
+ case OP_MOD_SUB_OVERWRITE:
+ case OP_MOD_SUB_VA_OVERWRITE:
+ case OP_MOD_SUB_DEL_ALL:
+ case OP_NET_KEY_ADD:
+ case OP_NET_KEY_UPDATE:
+ case OP_NET_KEY_DEL:
+ case OP_APP_KEY_ADD:
+ case OP_APP_KEY_UPDATE:
+ case OP_APP_KEY_DEL:
+ case OP_NODE_IDENTITY_SET:
+ case OP_MOD_APP_BIND:
+ case OP_MOD_APP_UNBIND:
+ case OP_NODE_RESET:
+ case OP_FRIEND_SET:
+ case OP_KRP_SET:
+ case OP_HEARTBEAT_PUB_SET:
+ case OP_HEARTBEAT_SUB_SET:
+ case OP_NET_TRANSMIT_SET:
+ evt_type = 0x01;
+ break;
+ default:
+ break;
+ }
+
+ bt_mesh_callback_config_status_to_btc(node->opcode, evt_type, model,
+ ctx, (const u8_t *)status, len);
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&data->queue, node);
+ }
+
+ switch (ctx->recv_op) {
+ case OP_DEV_COMP_DATA_STATUS: {
+ struct bt_mesh_cfg_comp_data_status *val;
+ val = (struct bt_mesh_cfg_comp_data_status *)status;
+ bt_mesh_free_buf(val->comp_data);
+ break;
+ }
+ case OP_MOD_SUB_LIST:
+ case OP_MOD_SUB_LIST_VND: {
+ struct bt_mesh_cfg_mod_sub_list *val = status;
+ bt_mesh_free_buf(val->addr);
+ break;
+ }
+ case OP_NET_KEY_LIST: {
+ struct bt_mesh_cfg_net_key_list *val = status;
+ bt_mesh_free_buf(val->net_idx);
+ break;
+ }
+ case OP_APP_KEY_LIST: {
+ struct bt_mesh_cfg_app_key_list *val = status;
+ bt_mesh_free_buf(val->app_idx);
+ break;
+ }
+ case OP_SIG_MOD_APP_LIST:
+ case OP_VND_MOD_APP_LIST: {
+ struct bt_mesh_cfg_mod_app_list *val = status;
+ bt_mesh_free_buf(val->app_idx);
+ break;
+ }
+ default:
+ break;
+ }
+}
+
+static void comp_data_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_comp_data_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.page = net_buf_simple_pull_u8(buf);
+ status.comp_data = bt_mesh_alloc_buf(buf->len);
+ if (!status.comp_data) {
+ BT_ERR("%s: allocate memory for comp_data fail", __func__);
+ return;
+ }
+
+ net_buf_simple_init(status.comp_data, 0);
+ net_buf_simple_add_mem(status.comp_data, buf->data, buf->len);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_comp_data_status));
+}
+
+static void state_status_u8(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t status = 0;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status = net_buf_simple_pull_u8(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(u8_t));
+}
+
+static void beacon_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ state_status_u8(model, ctx, buf);
+}
+
+static void ttl_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ state_status_u8(model, ctx, buf);
+}
+
+static void friend_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ state_status_u8(model, ctx, buf);
+}
+
+static void gatt_proxy_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ state_status_u8(model, ctx, buf);
+}
+
+static void relay_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_relay_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.relay = net_buf_simple_pull_u8(buf);
+ status.retransmit = net_buf_simple_pull_u8(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_relay_status));
+}
+
+static void net_key_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_netkey_status status = {0};
+ u16_t app_idx;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ key_idx_unpack(buf, &status.net_idx, &app_idx);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_netkey_status));
+}
+
+static void app_key_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_appkey_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ key_idx_unpack(buf, &status.net_idx, &status.app_idx);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_appkey_status));
+}
+
+static void mod_app_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_mod_app_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ status.elem_addr = net_buf_simple_pull_le16(buf);
+ status.app_idx = net_buf_simple_pull_le16(buf);
+ if (buf->len >= 4) {
+ status.cid = net_buf_simple_pull_le16(buf);
+ } else {
+ status.cid = CID_NVAL;
+ }
+ status.mod_id = net_buf_simple_pull_le16(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_mod_app_status));
+}
+
+static void mod_pub_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_mod_pub_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ status.elem_addr = net_buf_simple_pull_le16(buf);
+ status.addr = net_buf_simple_pull_le16(buf);
+ status.app_idx = net_buf_simple_pull_le16(buf);
+ status.cred_flag = (status.app_idx & BIT(12));
+ status.app_idx &= BIT_MASK(12);
+ status.ttl = net_buf_simple_pull_u8(buf);
+ status.period = net_buf_simple_pull_u8(buf);
+ status.transmit = net_buf_simple_pull_u8(buf);
+ if (buf->len >= 4) {
+ status.cid = net_buf_simple_pull_le16(buf);
+ } else {
+ status.cid = CID_NVAL;
+ }
+ status.mod_id = net_buf_simple_pull_le16(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_mod_pub_status));
+}
+
+static void mod_sub_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_mod_sub_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ status.elem_addr = net_buf_simple_pull_le16(buf);
+ status.sub_addr = net_buf_simple_pull_le16(buf);
+ if (buf->len >= 4) {
+ status.cid = net_buf_simple_pull_le16(buf);
+ } else {
+ status.cid = CID_NVAL;
+ }
+ status.mod_id = net_buf_simple_pull_le16(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_mod_sub_status));
+}
+
+static void hb_sub_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_hb_sub_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ status.src = net_buf_simple_pull_le16(buf);
+ status.dst = net_buf_simple_pull_le16(buf);
+ status.period = net_buf_simple_pull_u8(buf);
+ status.count = net_buf_simple_pull_u8(buf);
+ status.min = net_buf_simple_pull_u8(buf);
+ status.max = net_buf_simple_pull_u8(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_hb_sub_status));
+}
+
+static void hb_pub_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_hb_pub_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ status.dst = net_buf_simple_pull_le16(buf);
+ status.count = net_buf_simple_pull_u8(buf);
+ status.period = net_buf_simple_pull_u8(buf);
+ status.ttl = net_buf_simple_pull_u8(buf);
+ status.feat = net_buf_simple_pull_u8(buf);
+ status.net_idx = net_buf_simple_pull_u8(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_hb_sub_status));
+}
+
+static void node_reset_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ cfg_client_cancel(model, ctx, NULL, 0);
+}
+
+static void mod_sub_list(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_mod_sub_list list = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ list.status = net_buf_simple_pull_u8(buf);
+ list.elem_addr = net_buf_simple_pull_le16(buf);
+ if (ctx->recv_op == OP_MOD_SUB_LIST_VND) {
+ list.cid = net_buf_simple_pull_le16(buf);
+ } else {
+ list.cid = CID_NVAL;
+ }
+ list.mod_id = net_buf_simple_pull_le16(buf);
+
+ list.addr = bt_mesh_alloc_buf(buf->len);
+ if (!list.addr) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return;
+ }
+ net_buf_simple_init(list.addr, 0);
+ net_buf_simple_add_mem(list.addr, buf->data, buf->len);
+
+ cfg_client_cancel(model, ctx, &list, sizeof(struct bt_mesh_cfg_mod_sub_list));
+}
+
+static void net_key_list(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_net_key_list list = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ list.net_idx = bt_mesh_alloc_buf(buf->len);
+ if (!list.net_idx) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return;
+ }
+ net_buf_simple_init(list.net_idx, 0);
+ net_buf_simple_add_mem(list.net_idx, buf->data, buf->len);
+
+ cfg_client_cancel(model, ctx, &list, sizeof(struct bt_mesh_cfg_net_key_list));
+}
+
+static void app_key_list(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_app_key_list list = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ list.status = net_buf_simple_pull_u8(buf);
+ list.net_idx = net_buf_simple_pull_le16(buf);
+ list.app_idx = bt_mesh_alloc_buf(buf->len);
+ if (!list.app_idx) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return;
+ }
+ net_buf_simple_init(list.app_idx, 0);
+ net_buf_simple_add_mem(list.app_idx, buf->data, buf->len);
+
+ cfg_client_cancel(model, ctx, &list, sizeof(struct bt_mesh_cfg_app_key_list));
+}
+
+static void node_id_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_node_id_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ status.net_idx = net_buf_simple_pull_le16(buf);
+ status.identity = net_buf_simple_pull_u8(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_node_id_status));
+}
+
+static void mod_app_list(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_mod_app_list list = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ list.status = net_buf_simple_pull_u8(buf);
+ list.elem_addr = net_buf_simple_pull_le16(buf);
+ if (ctx->recv_op == OP_VND_MOD_APP_LIST) {
+ list.cid = net_buf_simple_pull_le16(buf);
+ } else {
+ list.cid = CID_NVAL;
+ }
+ list.mod_id = net_buf_simple_pull_le16(buf);
+
+ list.app_idx = bt_mesh_alloc_buf(buf->len);
+ if (!list.app_idx) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return;
+ }
+ net_buf_simple_init(list.app_idx, 0);
+ net_buf_simple_add_mem(list.app_idx, buf->data, buf->len);
+
+ cfg_client_cancel(model, ctx, &list, sizeof(struct bt_mesh_cfg_mod_app_list));
+}
+
+static void kr_phase_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_key_refresh_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.status = net_buf_simple_pull_u8(buf);
+ status.net_idx = net_buf_simple_pull_le16(buf);
+ status.phase = net_buf_simple_pull_u8(buf);
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_key_refresh_status));
+}
+
+static void lpn_pollto_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_lpn_pollto_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.lpn_addr = net_buf_simple_pull_le16(buf);
+ status.timeout = net_buf_simple_pull_u8(buf);
+ status.timeout |= net_buf_simple_pull_u8(buf) << 8;
+ status.timeout |= net_buf_simple_pull_u8(buf) << 16;
+
+ cfg_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_cfg_lpn_pollto_status));
+}
+
+static void net_trans_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ state_status_u8(model, ctx, buf);
+}
+
+const struct bt_mesh_model_op bt_mesh_cfg_cli_op[] = {
+ { OP_DEV_COMP_DATA_STATUS, 15, comp_data_status },
+ { OP_BEACON_STATUS, 1, beacon_status },
+ { OP_DEFAULT_TTL_STATUS, 1, ttl_status },
+ { OP_FRIEND_STATUS, 1, friend_status },
+ { OP_GATT_PROXY_STATUS, 1, gatt_proxy_status },
+ { OP_RELAY_STATUS, 2, relay_status },
+ { OP_NET_KEY_STATUS, 3, net_key_status },
+ { OP_APP_KEY_STATUS, 4, app_key_status },
+ { OP_MOD_APP_STATUS, 7, mod_app_status },
+ { OP_MOD_PUB_STATUS, 12, mod_pub_status },
+ { OP_MOD_SUB_STATUS, 7, mod_sub_status },
+ { OP_HEARTBEAT_SUB_STATUS, 9, hb_sub_status },
+ { OP_HEARTBEAT_PUB_STATUS, 10, hb_pub_status },
+ { OP_NODE_RESET_STATUS, 0, node_reset_status },
+ { OP_MOD_SUB_LIST, 5, mod_sub_list },
+ { OP_MOD_SUB_LIST_VND, 7, mod_sub_list },
+ { OP_NET_KEY_LIST, 2, net_key_list },
+ { OP_APP_KEY_LIST, 3, app_key_list },
+ { OP_NODE_IDENTITY_STATUS, 4, node_id_status },
+ { OP_SIG_MOD_APP_LIST, 5, mod_app_list },
+ { OP_VND_MOD_APP_LIST, 7, mod_app_list },
+ { OP_KRP_STATUS, 4, kr_phase_status },
+ { OP_LPN_TIMEOUT_STATUS, 5, lpn_pollto_status },
+ { OP_NET_TRANSMIT_STATUS, 1, net_trans_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+int bt_mesh_cfg_comp_data_get(struct bt_mesh_msg_ctx *ctx, u8_t page)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_DEV_COMP_DATA_GET);
+ net_buf_simple_add_u8(msg, page);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_DEV_COMP_DATA_GET, ctx,
+ msg, timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+static int get_state_u8(struct bt_mesh_msg_ctx *ctx, u32_t op)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ bt_mesh_model_msg_init(msg, op);
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+static int set_state_u8(struct bt_mesh_msg_ctx *ctx, u32_t op, u8_t new_val)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ int err;
+
+ bt_mesh_model_msg_init(msg, op);
+ net_buf_simple_add_u8(msg, new_val);
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_beacon_get(struct bt_mesh_msg_ctx *ctx)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return get_state_u8(ctx, OP_BEACON_GET);
+}
+
+int bt_mesh_cfg_beacon_set(struct bt_mesh_msg_ctx *ctx, u8_t val)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return set_state_u8(ctx, OP_BEACON_SET, val);
+}
+
+int bt_mesh_cfg_ttl_get(struct bt_mesh_msg_ctx *ctx)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return get_state_u8(ctx, OP_DEFAULT_TTL_GET);
+}
+
+int bt_mesh_cfg_ttl_set(struct bt_mesh_msg_ctx *ctx, u8_t val)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return set_state_u8(ctx, OP_DEFAULT_TTL_SET, val);
+}
+
+int bt_mesh_cfg_friend_get(struct bt_mesh_msg_ctx *ctx)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return get_state_u8(ctx, OP_FRIEND_GET);
+}
+
+int bt_mesh_cfg_friend_set(struct bt_mesh_msg_ctx *ctx, u8_t val)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return set_state_u8(ctx, OP_FRIEND_SET, val);
+}
+
+int bt_mesh_cfg_gatt_proxy_get(struct bt_mesh_msg_ctx *ctx)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return get_state_u8(ctx, OP_GATT_PROXY_GET);
+}
+
+int bt_mesh_cfg_gatt_proxy_set(struct bt_mesh_msg_ctx *ctx, u8_t val)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return set_state_u8(ctx, OP_GATT_PROXY_SET, val);
+}
+
+int bt_mesh_cfg_relay_get(struct bt_mesh_msg_ctx *ctx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_RELAY_GET);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_RELAY_GET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_relay_set(struct bt_mesh_msg_ctx *ctx, u8_t new_relay,
+ u8_t new_transmit)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_RELAY_SET);
+ net_buf_simple_add_u8(msg, new_relay);
+ net_buf_simple_add_u8(msg, new_transmit);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_RELAY_SET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_net_key_add(struct bt_mesh_msg_ctx *ctx, u16_t key_net_idx,
+ const u8_t net_key[16])
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 18 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !net_key) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NET_KEY_ADD);
+ net_buf_simple_add_le16(msg, key_net_idx);
+ net_buf_simple_add_mem(msg, net_key, 16);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_NET_KEY_ADD, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_app_key_add(struct bt_mesh_msg_ctx *ctx, u16_t key_net_idx,
+ u16_t key_app_idx, const u8_t app_key[16])
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(1 + 19 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !app_key) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_ADD);
+ key_idx_pack(msg, key_net_idx, key_app_idx);
+ net_buf_simple_add_mem(msg, app_key, 16);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_APP_KEY_ADD, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_app_bind(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_app_idx, u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 8 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_MOD_APP_BIND);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_le16(msg, mod_app_idx);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_MOD_APP_BIND, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+static int mod_sub(u32_t op, struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t sub_addr, u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 8 + 4);
+ int err;
+
+ bt_mesh_model_msg_init(msg, op);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_le16(msg, sub_addr);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_sub_add(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t sub_addr, u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return mod_sub(OP_MOD_SUB_ADD, ctx, elem_addr, sub_addr, mod_id, cid);
+}
+
+int bt_mesh_cfg_mod_sub_del(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t sub_addr, u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return mod_sub(OP_MOD_SUB_DEL, ctx, elem_addr, sub_addr, mod_id, cid);
+}
+
+int bt_mesh_cfg_mod_sub_overwrite(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t sub_addr, u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return mod_sub(OP_MOD_SUB_OVERWRITE, ctx, elem_addr, sub_addr, mod_id, cid);
+}
+
+static int mod_sub_va(u32_t op, struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ const u8_t label[16], u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 22 + 4);
+ int err;
+
+ BT_DBG("net_idx 0x%04x addr 0x%04x elem_addr 0x%04x label %s",
+ ctx->net_idx, ctx->addr, elem_addr, label);
+ BT_DBG("mod_id 0x%04x cid 0x%04x", mod_id, cid);
+
+ bt_mesh_model_msg_init(msg, op);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_mem(msg, label, 16);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_sub_va_add(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ const u8_t label[16], u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr || !label) {
+ return -EINVAL;
+ }
+ return mod_sub_va(OP_MOD_SUB_VA_ADD, ctx, elem_addr, label, mod_id, cid);
+}
+
+int bt_mesh_cfg_mod_sub_va_del(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ const u8_t label[16], u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr || !label) {
+ return -EINVAL;
+ }
+ return mod_sub_va(OP_MOD_SUB_VA_DEL, ctx, elem_addr, label, mod_id, cid);
+}
+
+int bt_mesh_cfg_mod_sub_va_overwrite(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ const u8_t label[16], u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr || !label) {
+ return -EINVAL;
+ }
+ return mod_sub_va(OP_MOD_SUB_VA_OVERWRITE, ctx, elem_addr, label, mod_id, cid);
+}
+
+int bt_mesh_cfg_mod_pub_get(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 6 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_MOD_PUB_GET);
+ net_buf_simple_add_le16(msg, elem_addr);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_MOD_PUB_GET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_pub_set(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid,
+ struct bt_mesh_cfg_mod_pub *pub)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 13 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !pub) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_MOD_PUB_SET);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_le16(msg, pub->addr);
+ net_buf_simple_add_le16(msg, (pub->app_idx | (pub->cred_flag << 12)));
+ net_buf_simple_add_u8(msg, pub->ttl);
+ net_buf_simple_add_u8(msg, pub->period);
+ net_buf_simple_add_u8(msg, pub->transmit);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_MOD_PUB_SET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_hb_sub_set(struct bt_mesh_msg_ctx *ctx,
+ struct bt_mesh_cfg_hb_sub *sub)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 5 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !sub) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEARTBEAT_SUB_SET);
+ net_buf_simple_add_le16(msg, sub->src);
+ net_buf_simple_add_le16(msg, sub->dst);
+ net_buf_simple_add_u8(msg, sub->period);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_HEARTBEAT_SUB_SET, ctx,
+ msg, timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_hb_sub_get(struct bt_mesh_msg_ctx *ctx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEARTBEAT_SUB_GET);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_HEARTBEAT_SUB_GET, ctx,
+ msg, timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_hb_pub_set(struct bt_mesh_msg_ctx *ctx,
+ const struct bt_mesh_cfg_hb_pub *pub)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 9 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !pub) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEARTBEAT_PUB_SET);
+ net_buf_simple_add_le16(msg, pub->dst);
+ net_buf_simple_add_u8(msg, pub->count);
+ net_buf_simple_add_u8(msg, pub->period);
+ net_buf_simple_add_u8(msg, pub->ttl);
+ net_buf_simple_add_le16(msg, pub->feat);
+ net_buf_simple_add_le16(msg, pub->net_idx);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_HEARTBEAT_PUB_SET, ctx,
+ msg, timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_hb_pub_get(struct bt_mesh_msg_ctx *ctx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEARTBEAT_PUB_GET);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_HEARTBEAT_PUB_GET, ctx,
+ msg, timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_node_reset(struct bt_mesh_msg_ctx *ctx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NODE_RESET);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_NODE_RESET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_pub_va_set(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid, const u8_t label[16],
+ struct bt_mesh_cfg_mod_pub *pub)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 27 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !label || !pub) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_MOD_PUB_VA_SET);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_mem(msg, label, 16);
+ net_buf_simple_add_le16(msg, (pub->app_idx | (pub->cred_flag << 12)));
+ net_buf_simple_add_u8(msg, pub->ttl);
+ net_buf_simple_add_u8(msg, pub->period);
+ net_buf_simple_add_u8(msg, pub->transmit);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_MOD_PUB_VA_SET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_sub_del_all(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 6 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_MOD_SUB_DEL_ALL);
+ net_buf_simple_add_le16(msg, elem_addr);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_MOD_SUB_DEL_ALL, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+static int mod_sub_get(u32_t op, struct bt_mesh_msg_ctx *ctx,
+ u16_t elem_addr, u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 6 + 4);
+ int err;
+
+ bt_mesh_model_msg_init(msg, op);
+ net_buf_simple_add_le16(msg, elem_addr);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_sub_get(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr, u16_t mod_id)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return mod_sub_get(OP_MOD_SUB_GET, ctx, elem_addr, mod_id, CID_NVAL);
+}
+
+int bt_mesh_cfg_mod_sub_get_vnd(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr || cid == CID_NVAL) {
+ return -EINVAL;
+ }
+ return mod_sub_get(OP_MOD_SUB_GET_VND, ctx, elem_addr, mod_id, cid);
+}
+
+int bt_mesh_cfg_net_key_update(struct bt_mesh_msg_ctx *ctx, u16_t net_idx,
+ const u8_t net_key[16])
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 18 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !net_key) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NET_KEY_UPDATE);
+ net_buf_simple_add_le16(msg, net_idx);
+ net_buf_simple_add_mem(msg, net_key, 16);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_NET_KEY_UPDATE, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_net_key_delete(struct bt_mesh_msg_ctx *ctx, u16_t net_idx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NET_KEY_DEL);
+ net_buf_simple_add_le16(msg, net_idx);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_NET_KEY_DEL, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_net_key_get(struct bt_mesh_msg_ctx *ctx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NET_KEY_GET);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_NET_KEY_GET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_app_key_update(struct bt_mesh_msg_ctx *ctx, u16_t net_idx,
+ u16_t app_idx, const u8_t app_key[16])
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(1 + 19 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr || !app_key) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_UPDATE);
+ key_idx_pack(msg, net_idx, app_idx);
+ net_buf_simple_add_mem(msg, app_key, 16);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_APP_KEY_UPDATE, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_app_key_delete(struct bt_mesh_msg_ctx *ctx, u16_t net_idx, u16_t app_idx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 3 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_DEL);
+ key_idx_pack(msg, net_idx, app_idx);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_APP_KEY_DEL, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_app_key_get(struct bt_mesh_msg_ctx *ctx, u16_t net_idx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_GET);
+ net_buf_simple_add_le16(msg, net_idx);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_APP_KEY_GET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+static int node_identity_op(u32_t op, struct bt_mesh_msg_ctx *ctx,
+ u16_t net_idx, u8_t identity)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 3 + 4);
+ int err;
+
+ bt_mesh_model_msg_init(msg, op);
+ net_buf_simple_add_le16(msg, net_idx);
+ if (op == OP_NODE_IDENTITY_SET) {
+ net_buf_simple_add_u8(msg, identity);
+ }
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_node_identity_get(struct bt_mesh_msg_ctx *ctx, u16_t net_idx)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return node_identity_op(OP_NODE_IDENTITY_GET, ctx, net_idx, 0xFF);
+}
+
+int bt_mesh_cfg_node_identity_set(struct bt_mesh_msg_ctx *ctx, u16_t net_idx, u8_t identity)
+{
+ if (!ctx || !ctx->addr || identity > 0x01) {
+ return -EINVAL;
+ }
+ return node_identity_op(OP_NODE_IDENTITY_SET, ctx, net_idx, identity);
+}
+
+int bt_mesh_cfg_mod_app_unbind(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t app_idx, u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 8 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_MOD_APP_UNBIND);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_le16(msg, app_idx);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_MOD_APP_UNBIND, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+static int mod_app_get(u32_t op, struct bt_mesh_msg_ctx *ctx,
+ u16_t elem_addr, u16_t mod_id, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 6 + 4);
+ int err;
+
+ bt_mesh_model_msg_init(msg, op);
+ net_buf_simple_add_le16(msg, elem_addr);
+ if (cid != CID_NVAL) {
+ net_buf_simple_add_le16(msg, cid);
+ }
+ net_buf_simple_add_le16(msg, mod_id);
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_mod_app_get(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr, u16_t mod_id)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return mod_app_get(OP_SIG_MOD_APP_GET, ctx, elem_addr, mod_id, CID_NVAL);
+}
+
+int bt_mesh_cfg_mod_app_get_vnd(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid)
+{
+ if (!ctx || !ctx->addr || cid == CID_NVAL) {
+ return -EINVAL;
+ }
+ return mod_app_get(OP_VND_MOD_APP_GET, ctx, elem_addr, mod_id, cid);
+}
+
+static int kr_phase_op(u32_t op, struct bt_mesh_msg_ctx *ctx,
+ u16_t net_idx, u8_t transition)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 3 + 4);
+ int err;
+
+ bt_mesh_model_msg_init(msg, op);
+ net_buf_simple_add_le16(msg, net_idx);
+ if (op == OP_KRP_SET) {
+ net_buf_simple_add_u8(msg, transition);
+ }
+
+ err = bt_mesh_client_send_msg(cli->model, op, ctx, msg, timeout_handler,
+ config_msg_timeout, true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_kr_phase_get(struct bt_mesh_msg_ctx *ctx, u16_t net_idx)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return kr_phase_op(OP_KRP_GET, ctx, net_idx, 0xFF);
+}
+
+int bt_mesh_cfg_kr_phase_set(struct bt_mesh_msg_ctx *ctx, u16_t net_idx, u8_t transition)
+{
+ if (!ctx || !ctx->addr || transition > 0x03) {
+ return -EINVAL;
+ }
+ return kr_phase_op(OP_KRP_SET, ctx, net_idx, transition);;
+}
+
+int bt_mesh_cfg_lpn_timeout_get(struct bt_mesh_msg_ctx *ctx, u16_t lpn_addr)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_LPN_TIMEOUT_GET);
+ net_buf_simple_add_le16(msg, lpn_addr);
+
+ err = bt_mesh_client_send_msg(cli->model, OP_LPN_TIMEOUT_GET, ctx, msg,
+ timeout_handler, config_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_cfg_net_transmit_get(struct bt_mesh_msg_ctx *ctx)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return get_state_u8(ctx, OP_NET_TRANSMIT_GET);
+}
+
+int bt_mesh_cfg_net_transmit_set(struct bt_mesh_msg_ctx *ctx, u8_t transmit)
+{
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+ return set_state_u8(ctx, OP_NET_TRANSMIT_SET, transmit);
+}
+
+s32_t bt_mesh_cfg_cli_timeout_get(void)
+{
+ return config_msg_timeout;
+}
+
+void bt_mesh_cfg_cli_timeout_set(s32_t timeout)
+{
+ config_msg_timeout = timeout;
+}
+
+int bt_mesh_cfg_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ bt_mesh_config_client_t *client = NULL;
+ config_internal_data_t *internal = NULL;
+
+ BT_DBG("primary %u", primary);
+
+ if (!primary) {
+ BT_ERR("Configuration Client only allowed in primary element");
+ return -EINVAL;
+ }
+
+ if (!model) {
+ BT_ERR("Configuration Client model is NULL");
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_config_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("No Configuration Client context provided");
+ return -EINVAL;
+ }
+
+ /* TODO: call osi_free() when deinit function is invoked*/
+ internal = osi_calloc(sizeof(config_internal_data_t));
+ if (!internal) {
+ BT_ERR("Allocate memory for Configuration Client internal data fail");
+ return -ENOMEM;
+ }
+
+ sys_slist_init(&internal->queue);
+
+ client->model = model;
+ client->op_pair_size = ARRAY_SIZE(cfg_op_pair);
+ client->op_pair = cfg_op_pair;
+ client->internal_data = internal;
+
+ cli = client;
+
+ /* Configuration Model security is device-key based */
+ model->keys[0] = BT_MESH_KEY_DEV;
+
+ return 0;
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/cfg_srv.c b/components/bt/ble_mesh/mesh_core/cfg_srv.c
new file mode 100644
index 0000000000..5b585af1b2
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/cfg_srv.c
@@ -0,0 +1,3465 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+#include "mesh_types.h"
+#include "mesh_util.h"
+#include "sdkconfig.h"
+#include "mesh_main.h"
+
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_MODEL)
+#include "mesh_trace.h"
+
+#include "cfg_srv.h"
+#include "mesh.h"
+#include "adv.h"
+#include "net.h"
+#include "lpn.h"
+#include "transport.h"
+#include "crypto.h"
+#include "access.h"
+#include "beacon.h"
+#include "proxy.h"
+#include "foundation.h"
+#include "friend.h"
+#include "settings.h"
+#include "common.h"
+#include "btc_ble_mesh_config_client.h"
+
+#define DEFAULT_TTL 7
+
+/* Maximum message length is 384 in BLE Mesh. Here for composition data,
+ * due to 1 octet opcode and 4 octets TransMIC, 379 octets can be used to
+ * store device composition data.
+ */
+#define COMP_DATA_MAX_LEN 379
+
+static struct bt_mesh_cfg_srv *conf;
+
+static struct label {
+ u16_t ref;
+ u16_t addr;
+ u8_t uuid[16];
+} labels[CONFIG_BT_MESH_LABEL_COUNT];
+
+static void hb_send(struct bt_mesh_model *model)
+{
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+ u16_t feat = 0;
+ struct __packed {
+ u8_t init_ttl;
+ u16_t feat;
+ } hb;
+ struct bt_mesh_msg_ctx ctx = {
+ .net_idx = cfg->hb_pub.net_idx,
+ .app_idx = BT_MESH_KEY_UNUSED,
+ .addr = cfg->hb_pub.dst,
+ .send_ttl = cfg->hb_pub.ttl,
+ };
+ struct bt_mesh_net_tx tx = {
+ .sub = bt_mesh_subnet_get(cfg->hb_pub.net_idx),
+ .ctx = &ctx,
+ .src = model->elem->addr,
+ .xmit = bt_mesh_net_transmit_get(),
+ };
+
+ hb.init_ttl = cfg->hb_pub.ttl;
+
+ if (bt_mesh_relay_get() == BT_MESH_RELAY_ENABLED) {
+ feat |= BT_MESH_FEAT_RELAY;
+ }
+
+ if (bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_ENABLED) {
+ feat |= BT_MESH_FEAT_PROXY;
+ }
+
+ if (bt_mesh_friend_get() == BT_MESH_FRIEND_ENABLED) {
+ feat |= BT_MESH_FEAT_FRIEND;
+ }
+
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+ if (bt_mesh.lpn.state != BT_MESH_LPN_DISABLED) {
+ feat |= BT_MESH_FEAT_LOW_POWER;
+ }
+#endif
+
+ hb.feat = sys_cpu_to_be16(feat);
+
+ BT_DBG("InitTTL %u feat 0x%04x", cfg->hb_pub.ttl, feat);
+
+ bt_mesh_ctl_send(&tx, TRANS_CTL_OP_HEARTBEAT, &hb, sizeof(hb),
+ NULL, NULL, NULL);
+}
+
+static int comp_add_elem(struct net_buf_simple *buf, struct bt_mesh_elem *elem,
+ bool primary)
+{
+ struct bt_mesh_model *mod;
+ int i;
+
+ if (net_buf_simple_tailroom(buf) <
+ 4 + (elem->model_count * 2) + (elem->vnd_model_count * 2)) {
+ BT_ERR("Too large device composition");
+ return -E2BIG;
+ }
+
+ net_buf_simple_add_le16(buf, elem->loc);
+
+ net_buf_simple_add_u8(buf, elem->model_count);
+ net_buf_simple_add_u8(buf, elem->vnd_model_count);
+
+ for (i = 0; i < elem->model_count; i++) {
+ mod = &elem->models[i];
+ net_buf_simple_add_le16(buf, mod->id);
+ }
+
+ for (i = 0; i < elem->vnd_model_count; i++) {
+ mod = &elem->vnd_models[i];
+ net_buf_simple_add_le16(buf, mod->vnd.company);
+ net_buf_simple_add_le16(buf, mod->vnd.id);
+ }
+
+ return 0;
+}
+
+static int comp_get_page_0(struct net_buf_simple *buf)
+{
+ u16_t feat = 0;
+ const struct bt_mesh_comp *comp;
+ int i;
+
+ comp = bt_mesh_comp_get();
+
+ if (IS_ENABLED(CONFIG_BT_MESH_RELAY)) {
+ feat |= BT_MESH_FEAT_RELAY;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ feat |= BT_MESH_FEAT_PROXY;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ feat |= BT_MESH_FEAT_FRIEND;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ feat |= BT_MESH_FEAT_LOW_POWER;
+ }
+
+ net_buf_simple_add_le16(buf, comp->cid);
+ net_buf_simple_add_le16(buf, comp->pid);
+ net_buf_simple_add_le16(buf, comp->vid);
+ net_buf_simple_add_le16(buf, CONFIG_BT_MESH_CRPL);
+ net_buf_simple_add_le16(buf, feat);
+
+ for (i = 0; i < comp->elem_count; i++) {
+ int err;
+
+ err = comp_add_elem(buf, &comp->elem[i], i == 0);
+ if (err) {
+ return err;
+ }
+ }
+
+ return 0;
+}
+
+static void dev_comp_data_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *sdu = NULL;
+ u8_t page;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ page = net_buf_simple_pull_u8(buf);
+ if (page != 0) {
+ BT_WARN("Composition page %u not available", page);
+ page = 0;
+ }
+
+ sdu = bt_mesh_alloc_buf(min(BT_MESH_TX_SDU_MAX, COMP_DATA_MAX_LEN));
+ if (!sdu) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return;
+ }
+
+ bt_mesh_model_msg_init(sdu, OP_DEV_COMP_DATA_STATUS);
+
+ net_buf_simple_add_u8(sdu, page);
+ if (comp_get_page_0(sdu) < 0) {
+ BT_ERR("Unable to get composition page 0");
+ bt_mesh_free_buf(sdu);
+ return;
+ }
+
+ if (bt_mesh_model_send(model, ctx, sdu, NULL, NULL)) {
+ BT_ERR("Unable to send Device Composition Status response");
+ }
+
+ bt_mesh_free_buf(sdu);
+ return;
+}
+
+static struct bt_mesh_model *get_model(struct bt_mesh_elem *elem,
+ struct net_buf_simple *buf, bool *vnd)
+{
+ if (buf->len < 4) {
+ u16_t id;
+
+ id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("ID 0x%04x addr 0x%04x", id, elem->addr);
+
+ *vnd = false;
+
+ return bt_mesh_model_find(elem, id);
+ } else {
+ u16_t company, id;
+
+ company = net_buf_simple_pull_le16(buf);
+ id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("Company 0x%04x ID 0x%04x addr 0x%04x", company, id,
+ elem->addr);
+
+ *vnd = true;
+
+ return bt_mesh_model_find_vnd(elem, company, id);
+ }
+}
+
+static bool app_key_is_valid(u16_t app_idx)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ struct bt_mesh_app_key *key = &bt_mesh.app_keys[i];
+
+ if (key->net_idx != BT_MESH_KEY_UNUSED &&
+ key->app_idx == app_idx) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+static u8_t _mod_pub_set(struct bt_mesh_model *model, u16_t pub_addr,
+ u16_t app_idx, u8_t cred_flag, u8_t ttl, u8_t period,
+ u8_t retransmit)
+{
+ if (!model->pub) {
+ return STATUS_NVAL_PUB_PARAM;
+ }
+
+ if (!IS_ENABLED(CONFIG_BT_MESH_LOW_POWER) && cred_flag) {
+ return STATUS_FEAT_NOT_SUPP;
+ }
+
+ if (!model->pub->update && period) {
+ return STATUS_NVAL_PUB_PARAM;
+ }
+
+ if (pub_addr == BT_MESH_ADDR_UNASSIGNED) {
+ model->pub->addr = BT_MESH_ADDR_UNASSIGNED;
+ model->pub->key = 0;
+ model->pub->cred = 0;
+ model->pub->ttl = 0;
+ model->pub->period = 0;
+ model->pub->retransmit = 0;
+ model->pub->count = 0;
+
+ if (model->pub->update) {
+ k_delayed_work_cancel(&model->pub->timer);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_pub(model);
+ }
+
+ return STATUS_SUCCESS;
+ }
+
+ if (!bt_mesh_app_key_find(app_idx)) {
+ return STATUS_INVALID_APPKEY;
+ }
+
+ model->pub->addr = pub_addr;
+ model->pub->key = app_idx;
+ model->pub->cred = cred_flag;
+ model->pub->ttl = ttl;
+ model->pub->period = period;
+ model->pub->retransmit = retransmit;
+
+ if (model->pub->update) {
+ s32_t period_ms;
+
+ period_ms = bt_mesh_model_pub_period_get(model);
+ BT_DBG("period %u ms", period_ms);
+
+ if (period_ms) {
+ k_delayed_work_submit(&model->pub->timer, period_ms);
+ } else {
+ k_delayed_work_cancel(&model->pub->timer);
+ }
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_pub(model);
+ }
+
+ return STATUS_SUCCESS;
+}
+
+static u8_t mod_bind(struct bt_mesh_model *model, u16_t key_idx)
+{
+ int i;
+
+ BT_DBG("key_idx 0x%04x", key_idx);
+
+ if (!app_key_is_valid(key_idx)) {
+ return STATUS_INVALID_APPKEY;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(model->keys); i++) {
+ /* Treat existing binding as success */
+ if (model->keys[i] == key_idx) {
+ return STATUS_SUCCESS;
+ }
+ }
+
+ for (i = 0; i < ARRAY_SIZE(model->keys); i++) {
+ if (model->keys[i] == BT_MESH_KEY_UNUSED) {
+ model->keys[i] = key_idx;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_bind(model);
+ }
+
+ return STATUS_SUCCESS;
+ }
+ }
+
+ return STATUS_INSUFF_RESOURCES;
+}
+
+static u8_t mod_unbind(struct bt_mesh_model *model, u16_t key_idx)
+{
+ int i;
+
+ BT_DBG("model %p key_idx 0x%04x", model, key_idx);
+
+ if (!app_key_is_valid(key_idx)) {
+ return STATUS_INVALID_APPKEY;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(model->keys); i++) {
+ if (model->keys[i] != key_idx) {
+ continue;
+ }
+
+ model->keys[i] = BT_MESH_KEY_UNUSED;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_bind(model);
+ }
+
+ if (model->pub && model->pub->key == key_idx) {
+ _mod_pub_set(model, BT_MESH_ADDR_UNASSIGNED,
+ 0, 0, 0, 0, 0);
+ }
+
+ return STATUS_SUCCESS;
+ }
+
+ return STATUS_CANNOT_BIND;
+}
+
+struct bt_mesh_app_key *bt_mesh_app_key_alloc(u16_t app_idx)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ struct bt_mesh_app_key *key = &bt_mesh.app_keys[i];
+
+ if (key->net_idx == BT_MESH_KEY_UNUSED) {
+ return key;
+ }
+ }
+
+ return NULL;
+}
+
+static u8_t app_key_set(u16_t net_idx, u16_t app_idx, const u8_t val[16],
+ bool update)
+{
+ struct bt_mesh_app_keys *keys;
+ struct bt_mesh_app_key *key;
+ struct bt_mesh_subnet *sub;
+
+ BT_DBG("net_idx 0x%04x app_idx %04x update %u val %s",
+ net_idx, app_idx, update, bt_hex(val, 16));
+
+ sub = bt_mesh_subnet_get(net_idx);
+ if (!sub) {
+ return STATUS_INVALID_NETKEY;
+ }
+
+ key = bt_mesh_app_key_find(app_idx);
+ if (update) {
+ if (!key) {
+ return STATUS_INVALID_APPKEY;
+ }
+
+ if (key->net_idx != net_idx) {
+ return STATUS_INVALID_BINDING;
+ }
+
+ keys = &key->keys[1];
+
+ /* The AppKey Update message shall generate an error when node
+ * is in normal operation, Phase 2, or Phase 3 or in Phase 1
+ * when the AppKey Update message on a valid AppKeyIndex when
+ * the AppKey value is different.
+ */
+ if (sub->kr_phase != BT_MESH_KR_PHASE_1) {
+ return STATUS_CANNOT_UPDATE;
+ }
+
+ if (key->updated) {
+ if (memcmp(keys->val, val, 16)) {
+ return STATUS_CANNOT_UPDATE;
+ } else {
+ return STATUS_SUCCESS;
+ }
+ }
+
+ key->updated = true;
+ } else {
+ if (key) {
+ if (key->net_idx == net_idx &&
+ !memcmp(key->keys[0].val, val, 16)) {
+ return STATUS_SUCCESS;
+ }
+
+ if (key->net_idx == net_idx) {
+ return STATUS_IDX_ALREADY_STORED;
+ } else {
+ return STATUS_INVALID_NETKEY;
+ }
+ }
+
+ key = bt_mesh_app_key_alloc(app_idx);
+ if (!key) {
+ return STATUS_INSUFF_RESOURCES;
+ }
+
+ keys = &key->keys[0];
+ }
+
+ if (bt_mesh_app_id(val, &keys->id)) {
+ if (update) {
+ key->updated = false;
+ }
+
+ return STATUS_STORAGE_FAIL;
+ }
+
+ BT_DBG("app_idx 0x%04x AID 0x%02x", app_idx, keys->id);
+
+ key->net_idx = net_idx;
+ key->app_idx = app_idx;
+ memcpy(keys->val, val, 16);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ BT_DBG("Storing AppKey persistently");
+ bt_mesh_store_app_key(key);
+ }
+
+ return STATUS_SUCCESS;
+}
+
+static void app_key_add(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 4 + 4);
+ u16_t key_net_idx, key_app_idx;
+ u8_t status;
+
+ key_idx_unpack(buf, &key_net_idx, &key_app_idx);
+
+ BT_DBG("AppIdx 0x%04x NetIdx 0x%04x", key_app_idx, key_net_idx);
+
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_STATUS);
+
+ status = app_key_set(key_net_idx, key_app_idx, buf->data, false);
+ BT_DBG("status 0x%02x", status);
+ net_buf_simple_add_u8(msg, status);
+
+ key_idx_pack(msg, key_net_idx, key_app_idx);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send App Key Status response");
+ return;
+ }
+
+#if defined(CONFIG_BT_MESH_FAST_PROV)
+ bt_mesh_callback_cfg_server_event_to_btc(0x0, model, ctx,
+ (u8_t *)&key_app_idx, sizeof(u16_t));
+#endif
+}
+
+static void app_key_update(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 4 + 4);
+ u16_t key_net_idx, key_app_idx;
+ u8_t status;
+
+ key_idx_unpack(buf, &key_net_idx, &key_app_idx);
+
+ BT_DBG("AppIdx 0x%04x NetIdx 0x%04x", key_app_idx, key_net_idx);
+
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_STATUS);
+
+ status = app_key_set(key_net_idx, key_app_idx, buf->data, true);
+ BT_DBG("status 0x%02x", status);
+ net_buf_simple_add_u8(msg, status);
+
+ key_idx_pack(msg, key_net_idx, key_app_idx);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send App Key Status response");
+ }
+}
+
+static void _mod_unbind(struct bt_mesh_model *mod, struct bt_mesh_elem *elem,
+ bool vnd, bool primary, void *user_data)
+{
+ u16_t *key_idx = user_data;
+
+ mod_unbind(mod, *key_idx);
+}
+
+void bt_mesh_app_key_del(struct bt_mesh_app_key *key)
+{
+ bt_mesh_model_foreach(_mod_unbind, &key->app_idx);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_clear_app_key(key);
+ }
+
+ key->net_idx = BT_MESH_KEY_UNUSED;
+ memset(key->keys, 0, sizeof(key->keys));
+}
+
+static void app_key_del(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 4 + 4);
+ u16_t key_net_idx, key_app_idx;
+ struct bt_mesh_app_key *key;
+ u8_t status;
+
+ key_idx_unpack(buf, &key_net_idx, &key_app_idx);
+
+ BT_DBG("AppIdx 0x%04x NetIdx 0x%04x", key_app_idx, key_net_idx);
+
+ if (!bt_mesh_subnet_get(key_net_idx)) {
+ status = STATUS_INVALID_NETKEY;
+ goto send_status;
+ }
+
+ key = bt_mesh_app_key_find(key_app_idx);
+ if (!key) {
+ /* Treat as success, since the client might have missed a
+ * previous response and is resending the request.
+ */
+ status = STATUS_SUCCESS;
+ goto send_status;
+ }
+
+ if (key->net_idx != key_net_idx) {
+ status = STATUS_INVALID_BINDING;
+ goto send_status;
+ }
+
+ bt_mesh_app_key_del(key);
+ status = STATUS_SUCCESS;
+
+send_status:
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+
+ key_idx_pack(msg, key_net_idx, key_app_idx);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send App Key Status response");
+ }
+}
+
+/* Index list length: 3 bytes for every pair and 2 bytes for an odd idx */
+#define IDX_LEN(num) (((num) / 2) * 3 + ((num) % 2) * 2)
+
+static void app_key_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg =
+ NET_BUF_SIMPLE(2 + 3 + 4 +
+ IDX_LEN(CONFIG_BT_MESH_APP_KEY_COUNT));
+ u16_t get_idx, i, prev;
+ u8_t status;
+
+ get_idx = net_buf_simple_pull_le16(buf);
+ if (get_idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", get_idx);
+ return;
+ }
+
+ BT_DBG("idx 0x%04x", get_idx);
+
+ bt_mesh_model_msg_init(msg, OP_APP_KEY_LIST);
+
+ if (!bt_mesh_subnet_get(get_idx)) {
+ status = STATUS_INVALID_NETKEY;
+ } else {
+ status = STATUS_SUCCESS;
+ }
+
+ net_buf_simple_add_u8(msg, status);
+ net_buf_simple_add_le16(msg, get_idx);
+
+ if (status != STATUS_SUCCESS) {
+ goto send_status;
+ }
+
+ prev = BT_MESH_KEY_UNUSED;
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ struct bt_mesh_app_key *key = &bt_mesh.app_keys[i];
+
+ if (key->net_idx != get_idx) {
+ continue;
+ }
+
+ if (prev == BT_MESH_KEY_UNUSED) {
+ prev = key->app_idx;
+ continue;
+ }
+
+ key_idx_pack(msg, prev, key->app_idx);
+ prev = BT_MESH_KEY_UNUSED;
+ }
+
+ if (prev != BT_MESH_KEY_UNUSED) {
+ net_buf_simple_add_le16(msg, prev);
+ }
+
+send_status:
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send AppKey List");
+ }
+}
+
+static void beacon_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ bt_mesh_model_msg_init(msg, OP_BEACON_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_beacon_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Config Beacon Status response");
+ }
+}
+
+static void beacon_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ if (!cfg) {
+ BT_WARN("No Configuration Server context available");
+ } else if (buf->data[0] == 0x00 || buf->data[0] == 0x01) {
+ if (buf->data[0] != cfg->beacon) {
+ cfg->beacon = buf->data[0];
+#if CONFIG_BT_MESH_NODE
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_cfg();
+ }
+
+ if (cfg->beacon) {
+ bt_mesh_beacon_enable();
+ } else {
+ bt_mesh_beacon_disable();
+ }
+#endif
+ }
+ } else {
+ BT_WARN("Invalid Config Beacon value 0x%02x", buf->data[0]);
+ return;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_BEACON_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_beacon_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Config Beacon Status response");
+ }
+}
+
+static void default_ttl_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ bt_mesh_model_msg_init(msg, OP_DEFAULT_TTL_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_default_ttl_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Default TTL Status response");
+ }
+}
+
+static void default_ttl_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ if (!cfg) {
+ BT_WARN("No Configuration Server context available");
+ } else if (buf->data[0] <= BT_MESH_TTL_MAX && buf->data[0] != 0x01) {
+ if (cfg->default_ttl != buf->data[0]) {
+ cfg->default_ttl = buf->data[0];
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_cfg();
+ }
+ }
+ } else {
+ BT_WARN("Prohibited Default TTL value 0x%02x", buf->data[0]);
+ return;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_DEFAULT_TTL_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_default_ttl_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Default TTL Status response");
+ }
+}
+
+static void send_gatt_proxy_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+
+ bt_mesh_model_msg_init(msg, OP_GATT_PROXY_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_gatt_proxy_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send GATT Proxy Status");
+ }
+}
+
+static void gatt_proxy_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ send_gatt_proxy_status(model, ctx);
+}
+
+static void gatt_proxy_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+ struct bt_mesh_subnet *sub;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ if (buf->data[0] != 0x00 && buf->data[0] != 0x01) {
+ BT_WARN("Invalid GATT Proxy value 0x%02x", buf->data[0]);
+ return;
+ }
+
+ if (!IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) ||
+ bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_NOT_SUPPORTED) {
+ goto send_status;
+ }
+
+ if (!cfg) {
+ BT_WARN("No Configuration Server context available");
+ goto send_status;
+ }
+
+ BT_DBG("GATT Proxy 0x%02x -> 0x%02x", cfg->gatt_proxy, buf->data[0]);
+
+ if (cfg->gatt_proxy == buf->data[0]) {
+ goto send_status;
+ }
+
+ cfg->gatt_proxy = buf->data[0];
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_cfg();
+ }
+
+ if (cfg->gatt_proxy == BT_MESH_GATT_PROXY_DISABLED) {
+ int i;
+
+ /* Section 4.2.11.1: "When the GATT Proxy state is set to
+ * 0x00, the Node Identity state for all subnets shall be set
+ * to 0x00 and shall not be changed."
+ */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx != BT_MESH_KEY_UNUSED) {
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_proxy_identity_stop(sub);
+#endif
+ }
+ }
+
+ /* Section 4.2.11: "Upon transition from GATT Proxy state 0x01
+ * to GATT Proxy state 0x00 the GATT Bearer Server shall
+ * disconnect all GATT Bearer Clients.
+ */
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_proxy_gatt_disconnect();
+#endif
+ }
+
+ bt_mesh_adv_update();
+
+ sub = bt_mesh_subnet_get(cfg->hb_pub.net_idx);
+ if ((cfg->hb_pub.feat & BT_MESH_FEAT_PROXY) && sub) {
+ hb_send(model);
+ }
+
+send_status:
+ send_gatt_proxy_status(model, ctx);
+}
+
+static void net_transmit_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ bt_mesh_model_msg_init(msg, OP_NET_TRANSMIT_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_net_transmit_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Config Network Transmit Status");
+ }
+}
+
+static void net_transmit_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ BT_DBG("Transmit 0x%02x (count %u interval %ums)", buf->data[0],
+ BT_MESH_TRANSMIT_COUNT(buf->data[0]),
+ BT_MESH_TRANSMIT_INT(buf->data[0]));
+
+ if (!cfg) {
+ BT_WARN("No Configuration Server context available");
+ } else {
+ cfg->net_transmit = buf->data[0];
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_cfg();
+ }
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NET_TRANSMIT_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_net_transmit_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Network Transmit Status");
+ }
+}
+
+static void relay_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ bt_mesh_model_msg_init(msg, OP_RELAY_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_relay_get());
+ net_buf_simple_add_u8(msg, bt_mesh_relay_retransmit_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Config Relay Status response");
+ }
+}
+
+static void relay_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ if (!cfg) {
+ BT_WARN("No Configuration Server context available");
+ } else if (buf->data[0] == 0x00 || buf->data[0] == 0x01) {
+ struct bt_mesh_subnet *sub;
+ bool change;
+
+ if (cfg->relay == BT_MESH_RELAY_NOT_SUPPORTED) {
+ change = false;
+ } else {
+ change = (cfg->relay != buf->data[0]);
+ cfg->relay = buf->data[0];
+ cfg->relay_retransmit = buf->data[1];
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_cfg();
+ }
+ }
+
+ BT_DBG("Relay 0x%02x (%s) xmit 0x%02x (count %u interval %u)",
+ cfg->relay, change ? "changed" : "not changed",
+ cfg->relay_retransmit,
+ BT_MESH_TRANSMIT_COUNT(cfg->relay_retransmit),
+ BT_MESH_TRANSMIT_INT(cfg->relay_retransmit));
+
+ sub = bt_mesh_subnet_get(cfg->hb_pub.net_idx);
+ if ((cfg->hb_pub.feat & BT_MESH_FEAT_RELAY) && sub && change) {
+ hb_send(model);
+ }
+ } else {
+ BT_WARN("Invalid Relay value 0x%02x", buf->data[0]);
+ return;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_RELAY_STATUS);
+ net_buf_simple_add_u8(msg, bt_mesh_relay_get());
+ net_buf_simple_add_u8(msg, bt_mesh_relay_retransmit_get());
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Relay Status response");
+ }
+}
+
+static void send_mod_pub_status(struct bt_mesh_model *cfg_mod,
+ struct bt_mesh_msg_ctx *ctx,
+ u16_t elem_addr, u16_t pub_addr,
+ bool vnd, struct bt_mesh_model *mod,
+ u8_t status, u8_t *mod_id)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 14 + 4);
+
+ bt_mesh_model_msg_init(msg, OP_MOD_PUB_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+ net_buf_simple_add_le16(msg, elem_addr);
+
+ if (status != STATUS_SUCCESS) {
+ memset(net_buf_simple_add(msg, 7), 0, 7);
+ } else {
+ u16_t idx_cred;
+
+ net_buf_simple_add_le16(msg, pub_addr);
+
+ idx_cred = mod->pub->key | (u16_t)mod->pub->cred << 12;
+ net_buf_simple_add_le16(msg, idx_cred);
+ net_buf_simple_add_u8(msg, mod->pub->ttl);
+ net_buf_simple_add_u8(msg, mod->pub->period);
+ net_buf_simple_add_u8(msg, mod->pub->retransmit);
+ }
+
+ if (vnd) {
+ memcpy(net_buf_simple_add(msg, 4), mod_id, 4);
+ } else {
+ memcpy(net_buf_simple_add(msg, 2), mod_id, 2);
+ }
+
+ if (bt_mesh_model_send(cfg_mod, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Model Publication Status");
+ }
+}
+
+static void mod_pub_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u16_t elem_addr, pub_addr = 0;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id, status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ mod_id = buf->data;
+
+ BT_DBG("elem_addr 0x%04x", elem_addr);
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ if (!mod->pub) {
+ status = STATUS_NVAL_PUB_PARAM;
+ goto send_status;
+ }
+
+ pub_addr = mod->pub->addr;
+ status = STATUS_SUCCESS;
+
+send_status:
+ send_mod_pub_status(model, ctx, elem_addr, pub_addr, vnd, mod,
+ status, mod_id);
+}
+
+static void mod_pub_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t retransmit, status, pub_ttl, pub_period, cred_flag;
+ u16_t elem_addr, pub_addr, pub_app_idx;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ pub_addr = net_buf_simple_pull_le16(buf);
+ pub_app_idx = net_buf_simple_pull_le16(buf);
+ cred_flag = ((pub_app_idx >> 12) & BIT_MASK(1));
+ pub_app_idx &= BIT_MASK(12);
+
+ pub_ttl = net_buf_simple_pull_u8(buf);
+ if (pub_ttl > BT_MESH_TTL_MAX && pub_ttl != BT_MESH_TTL_DEFAULT) {
+ BT_ERR("Invalid TTL value 0x%02x", pub_ttl);
+ return;
+ }
+
+ pub_period = net_buf_simple_pull_u8(buf);
+ retransmit = net_buf_simple_pull_u8(buf);
+ mod_id = buf->data;
+
+ BT_DBG("elem_addr 0x%04x pub_addr 0x%04x cred_flag %u",
+ elem_addr, pub_addr, cred_flag);
+ BT_DBG("pub_app_idx 0x%03x, pub_ttl %u pub_period 0x%02x",
+ pub_app_idx, pub_ttl, pub_period);
+ BT_DBG("retransmit 0x%02x (count %u interval %ums)", retransmit,
+ BT_MESH_PUB_TRANSMIT_COUNT(retransmit),
+ BT_MESH_PUB_TRANSMIT_INT(retransmit));
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = _mod_pub_set(mod, pub_addr, pub_app_idx, cred_flag, pub_ttl,
+ pub_period, retransmit);
+
+send_status:
+ send_mod_pub_status(model, ctx, elem_addr, pub_addr, vnd, mod,
+ status, mod_id);
+}
+
+#if CONFIG_BT_MESH_LABEL_COUNT > 0
+static u8_t va_add(u8_t *label_uuid, u16_t *addr)
+{
+ struct label *free_slot = NULL;
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(labels); i++) {
+ if (!labels[i].ref) {
+ free_slot = &labels[i];
+ continue;
+ }
+
+ if (!memcmp(labels[i].uuid, label_uuid, 16)) {
+ *addr = labels[i].addr;
+ labels[i].ref++;
+ return STATUS_SUCCESS;
+ }
+ }
+
+ if (!free_slot) {
+ return STATUS_INSUFF_RESOURCES;
+ }
+
+ if (bt_mesh_virtual_addr(label_uuid, addr) < 0) {
+ return STATUS_UNSPECIFIED;
+ }
+
+ free_slot->ref = 1;
+ free_slot->addr = *addr;
+ memcpy(free_slot->uuid, label_uuid, 16);
+
+ return STATUS_SUCCESS;
+}
+
+static u8_t va_del(u8_t *label_uuid, u16_t *addr)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(labels); i++) {
+ if (!memcmp(labels[i].uuid, label_uuid, 16)) {
+ if (addr) {
+ *addr = labels[i].addr;
+ }
+
+ labels[i].ref--;
+ return STATUS_SUCCESS;
+ }
+ }
+
+ if (addr) {
+ *addr = BT_MESH_ADDR_UNASSIGNED;
+ }
+
+ return STATUS_CANNOT_REMOVE;
+}
+
+static void mod_sub_list_clear(struct bt_mesh_model *mod)
+{
+ u8_t *label_uuid;
+ int i;
+
+ /* Unref stored labels related to this model */
+ for (i = 0; i < ARRAY_SIZE(mod->groups); i++) {
+ if (!BT_MESH_ADDR_IS_VIRTUAL(mod->groups[i])) {
+ continue;
+ }
+
+ label_uuid = bt_mesh_label_uuid_get(mod->groups[i]);
+ if (!label_uuid) {
+ BT_ERR("Label UUID not found");
+ continue;
+ }
+
+ va_del(label_uuid, NULL);
+ }
+
+ /* Clear all subscriptions (0x0000 is the unassigned address) */
+ memset(mod->groups, 0, sizeof(mod->groups));
+}
+
+static void mod_pub_va_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t retransmit, status, pub_ttl, pub_period, cred_flag;
+ u16_t elem_addr, pub_addr, pub_app_idx;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *label_uuid;
+ u8_t *mod_id;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ label_uuid = buf->data;
+ net_buf_simple_pull(buf, 16);
+
+ pub_app_idx = net_buf_simple_pull_le16(buf);
+ cred_flag = ((pub_app_idx >> 12) & BIT_MASK(1));
+ pub_app_idx &= BIT_MASK(12);
+ pub_ttl = net_buf_simple_pull_u8(buf);
+ if (pub_ttl > BT_MESH_TTL_MAX && pub_ttl != BT_MESH_TTL_DEFAULT) {
+ BT_ERR("Invalid TTL value 0x%02x", pub_ttl);
+ return;
+ }
+
+ pub_period = net_buf_simple_pull_u8(buf);
+ retransmit = net_buf_simple_pull_u8(buf);
+ mod_id = buf->data;
+
+ BT_DBG("elem_addr 0x%04x pub_addr 0x%04x cred_flag %u",
+ elem_addr, pub_addr, cred_flag);
+ BT_DBG("pub_app_idx 0x%03x, pub_ttl %u pub_period 0x%02x",
+ pub_app_idx, pub_ttl, pub_period);
+ BT_DBG("retransmit 0x%02x (count %u interval %ums)", retransmit,
+ BT_MESH_PUB_TRANSMIT_COUNT(retransmit),
+ BT_MESH_PUB_TRANSMIT_INT(retransmit));
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ pub_addr = 0;
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ pub_addr = 0;
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = va_add(label_uuid, &pub_addr);
+ if (status == STATUS_SUCCESS) {
+ status = _mod_pub_set(mod, pub_addr, pub_app_idx, cred_flag,
+ pub_ttl, pub_period, retransmit);
+ }
+
+send_status:
+ send_mod_pub_status(model, ctx, elem_addr, pub_addr, vnd, mod,
+ status, mod_id);
+}
+#else
+static void mod_sub_list_clear(struct bt_mesh_model *mod)
+{
+ /* Clear all subscriptions (0x0000 is the unassigned address) */
+ memset(mod->groups, 0, sizeof(mod->groups));
+}
+
+static void mod_pub_va_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t *mod_id, status;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u16_t elem_addr, pub_addr = 0;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ net_buf_simple_pull(buf, 16);
+ mod_id = net_buf_simple_pull(buf, 4);
+
+ BT_DBG("elem_addr 0x%04x", elem_addr);
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ if (!mod->pub) {
+ status = STATUS_NVAL_PUB_PARAM;
+ goto send_status;
+ }
+
+ pub_addr = mod->pub->addr;
+ status = STATUS_INSUFF_RESOURCES;
+
+send_status:
+ send_mod_pub_status(model, ctx, elem_addr, pub_addr, vnd, mod,
+ status, mod_id);
+}
+#endif /* CONFIG_BT_MESH_LABEL_COUNT > 0 */
+
+static void send_mod_sub_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, u8_t status,
+ u16_t elem_addr, u16_t sub_addr, u8_t *mod_id,
+ bool vnd)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 9 + 4);
+
+ BT_DBG("status 0x%02x elem_addr 0x%04x sub_addr 0x%04x", status,
+ elem_addr, sub_addr);
+
+ bt_mesh_model_msg_init(msg, OP_MOD_SUB_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_le16(msg, sub_addr);
+
+ if (vnd) {
+ memcpy(net_buf_simple_add(msg, 4), mod_id, 4);
+ } else {
+ memcpy(net_buf_simple_add(msg, 2), mod_id, 2);
+ }
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Model Subscription Status");
+ }
+}
+
+static void mod_sub_add(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u16_t elem_addr, sub_addr;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+ int i;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ sub_addr = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("elem_addr 0x%04x, sub_addr 0x%04x", elem_addr, sub_addr);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ if (!BT_MESH_ADDR_IS_GROUP(sub_addr)) {
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ if (bt_mesh_model_find_group(mod, sub_addr)) {
+ /* Tried to add existing subscription */
+ status = STATUS_SUCCESS;
+ goto send_status;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(mod->groups); i++) {
+ if (mod->groups[i] == BT_MESH_ADDR_UNASSIGNED) {
+ mod->groups[i] = sub_addr;
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(mod->groups)) {
+ status = STATUS_INSUFF_RESOURCES;
+ } else {
+ status = STATUS_SUCCESS;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_add(sub_addr);
+ }
+ }
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr, sub_addr,
+ mod_id, vnd);
+}
+
+static void mod_sub_del(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u16_t elem_addr, sub_addr;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id;
+ u16_t *match;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ sub_addr = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("elem_addr 0x%04x sub_addr 0x%04x", elem_addr, sub_addr);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ if (!BT_MESH_ADDR_IS_GROUP(sub_addr)) {
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ /* An attempt to remove a non-existing address shall be treated
+ * as a success.
+ */
+ status = STATUS_SUCCESS;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_del(&sub_addr, 1);
+ }
+
+ match = bt_mesh_model_find_group(mod, sub_addr);
+ if (match) {
+ *match = BT_MESH_ADDR_UNASSIGNED;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+ }
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr, sub_addr,
+ mod_id, vnd);
+}
+
+static void mod_sub_overwrite(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u16_t elem_addr, sub_addr;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ sub_addr = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("elem_addr 0x%04x sub_addr 0x%04x", elem_addr, sub_addr);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ if (!BT_MESH_ADDR_IS_GROUP(sub_addr)) {
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_del(mod->groups, ARRAY_SIZE(mod->groups));
+ }
+
+ mod_sub_list_clear(mod);
+
+ if (ARRAY_SIZE(mod->groups) > 0) {
+ mod->groups[0] = sub_addr;
+ status = STATUS_SUCCESS;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_add(sub_addr);
+ }
+ } else {
+ status = STATUS_INSUFF_RESOURCES;
+ }
+
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr, sub_addr,
+ mod_id, vnd);
+}
+
+static void mod_sub_del_all(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u16_t elem_addr;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("elem_addr 0x%04x", elem_addr);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_del(mod->groups, ARRAY_SIZE(mod->groups));
+ }
+
+ mod_sub_list_clear(mod);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+
+ status = STATUS_SUCCESS;
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr,
+ BT_MESH_ADDR_UNASSIGNED, mod_id, vnd);
+}
+
+static void mod_sub_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg =
+ NET_BUF_SIMPLE(2 + 5 + 4 +
+ CONFIG_BT_MESH_MODEL_GROUP_COUNT * 2);
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u16_t addr, id;
+ int i;
+
+ addr = net_buf_simple_pull_le16(buf);
+ id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("addr 0x%04x id 0x%04x", addr, id);
+
+ bt_mesh_model_msg_init(msg, OP_MOD_SUB_LIST);
+
+ elem = bt_mesh_elem_find(addr);
+ if (!elem) {
+ net_buf_simple_add_u8(msg, STATUS_INVALID_ADDRESS);
+ net_buf_simple_add_le16(msg, addr);
+ net_buf_simple_add_le16(msg, id);
+ goto send_list;
+ }
+
+ mod = bt_mesh_model_find(elem, id);
+ if (!mod) {
+ net_buf_simple_add_u8(msg, STATUS_INVALID_MODEL);
+ net_buf_simple_add_le16(msg, addr);
+ net_buf_simple_add_le16(msg, id);
+ goto send_list;
+ }
+
+ net_buf_simple_add_u8(msg, STATUS_SUCCESS);
+
+ net_buf_simple_add_le16(msg, addr);
+ net_buf_simple_add_le16(msg, id);
+
+ for (i = 0; i < ARRAY_SIZE(mod->groups); i++) {
+ if (mod->groups[i] != BT_MESH_ADDR_UNASSIGNED) {
+ net_buf_simple_add_le16(msg, mod->groups[i]);
+ }
+ }
+
+send_list:
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Model Subscription List");
+ }
+}
+
+static void mod_sub_get_vnd(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg =
+ NET_BUF_SIMPLE(2 + 7 + 4 +
+ CONFIG_BT_MESH_MODEL_GROUP_COUNT * 2);
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u16_t company, addr, id;
+ int i;
+
+ addr = net_buf_simple_pull_le16(buf);
+ company = net_buf_simple_pull_le16(buf);
+ id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("addr 0x%04x company 0x%04x id 0x%04x", addr, company, id);
+
+ bt_mesh_model_msg_init(msg, OP_MOD_SUB_LIST_VND);
+
+ elem = bt_mesh_elem_find(addr);
+ if (!elem) {
+ net_buf_simple_add_u8(msg, STATUS_INVALID_ADDRESS);
+ net_buf_simple_add_le16(msg, addr);
+ net_buf_simple_add_le16(msg, company);
+ net_buf_simple_add_le16(msg, id);
+ goto send_list;
+ }
+
+ mod = bt_mesh_model_find_vnd(elem, company, id);
+ if (!mod) {
+ net_buf_simple_add_u8(msg, STATUS_INVALID_MODEL);
+ net_buf_simple_add_le16(msg, addr);
+ net_buf_simple_add_le16(msg, company);
+ net_buf_simple_add_le16(msg, id);
+ goto send_list;
+ }
+
+ net_buf_simple_add_u8(msg, STATUS_SUCCESS);
+
+ net_buf_simple_add_le16(msg, addr);
+ net_buf_simple_add_le16(msg, company);
+ net_buf_simple_add_le16(msg, id);
+
+ for (i = 0; i < ARRAY_SIZE(mod->groups); i++) {
+ if (mod->groups[i] != BT_MESH_ADDR_UNASSIGNED) {
+ net_buf_simple_add_le16(msg, mod->groups[i]);
+ }
+ }
+
+send_list:
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Vendor Model Subscription List");
+ }
+}
+
+#if CONFIG_BT_MESH_LABEL_COUNT > 0
+static void mod_sub_va_add(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u16_t elem_addr, sub_addr;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *label_uuid;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+ int i;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ label_uuid = buf->data;
+ net_buf_simple_pull(buf, 16);
+
+ BT_DBG("elem_addr 0x%04x", elem_addr);
+
+ mod_id = buf->data;
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ sub_addr = BT_MESH_ADDR_UNASSIGNED;
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ sub_addr = BT_MESH_ADDR_UNASSIGNED;
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = va_add(label_uuid, &sub_addr);
+ if (status != STATUS_SUCCESS) {
+ goto send_status;
+ }
+
+ if (bt_mesh_model_find_group(mod, sub_addr)) {
+ /* Tried to add existing subscription */
+ status = STATUS_SUCCESS;
+ goto send_status;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(mod->groups); i++) {
+ if (mod->groups[i] == BT_MESH_ADDR_UNASSIGNED) {
+ mod->groups[i] = sub_addr;
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(mod->groups)) {
+ status = STATUS_INSUFF_RESOURCES;
+ } else {
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_add(sub_addr);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+
+ status = STATUS_SUCCESS;
+ }
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr, sub_addr,
+ mod_id, vnd);
+}
+
+static void mod_sub_va_del(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u16_t elem_addr, sub_addr;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *label_uuid;
+ u8_t *mod_id;
+ u16_t *match;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ label_uuid = buf->data;
+ net_buf_simple_pull(buf, 16);
+
+ BT_DBG("elem_addr 0x%04x", elem_addr);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ sub_addr = BT_MESH_ADDR_UNASSIGNED;
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ sub_addr = BT_MESH_ADDR_UNASSIGNED;
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = va_del(label_uuid, &sub_addr);
+ if (sub_addr == BT_MESH_ADDR_UNASSIGNED) {
+ goto send_status;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_del(&sub_addr, 1);
+ }
+
+ match = bt_mesh_model_find_group(mod, sub_addr);
+ if (match) {
+ *match = BT_MESH_ADDR_UNASSIGNED;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+
+ status = STATUS_SUCCESS;
+ } else {
+ status = STATUS_CANNOT_REMOVE;
+ }
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr, sub_addr,
+ mod_id, vnd);
+}
+
+static void mod_sub_va_overwrite(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u16_t elem_addr, sub_addr = BT_MESH_ADDR_UNASSIGNED;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *label_uuid;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ label_uuid = buf->data;
+ net_buf_simple_pull(buf, 16);
+
+ BT_DBG("elem_addr 0x%04x", elem_addr);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_del(mod->groups, ARRAY_SIZE(mod->groups));
+ }
+
+ mod_sub_list_clear(mod);
+
+ if (ARRAY_SIZE(mod->groups) > 0) {
+ status = va_add(label_uuid, &sub_addr);
+ if (status == STATUS_SUCCESS) {
+ mod->groups[0] = sub_addr;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_group_add(sub_addr);
+ }
+ }
+ } else {
+ status = STATUS_INSUFF_RESOURCES;
+ }
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr, sub_addr,
+ mod_id, vnd);
+}
+#else
+static void mod_sub_va_add(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u16_t elem_addr;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ net_buf_simple_pull(buf, 16);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = STATUS_INSUFF_RESOURCES;
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr,
+ BT_MESH_ADDR_UNASSIGNED, mod_id, vnd);
+}
+
+static void mod_sub_va_del(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_elem *elem;
+ u16_t elem_addr;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ net_buf_simple_pull(buf, 16);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ if (!get_model(elem, buf, &vnd)) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = STATUS_INSUFF_RESOURCES;
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr,
+ BT_MESH_ADDR_UNASSIGNED, mod_id, vnd);
+}
+
+static void mod_sub_va_overwrite(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_elem *elem;
+ u16_t elem_addr;
+ u8_t *mod_id;
+ u8_t status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ net_buf_simple_pull(buf, 18);
+
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ if (!get_model(elem, buf, &vnd)) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = STATUS_INSUFF_RESOURCES;
+
+send_status:
+ send_mod_sub_status(model, ctx, status, elem_addr,
+ BT_MESH_ADDR_UNASSIGNED, mod_id, vnd);
+}
+#endif /* CONFIG_BT_MESH_LABEL_COUNT > 0 */
+
+static void send_net_key_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ u16_t idx, u8_t status)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 3 + 4);
+
+ bt_mesh_model_msg_init(msg, OP_NET_KEY_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+ net_buf_simple_add_le16(msg, idx);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send NetKey Status");
+ }
+}
+
+static void net_key_add(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_subnet *sub;
+ u16_t idx;
+ int err;
+
+ idx = net_buf_simple_pull_le16(buf);
+ if (idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", idx);
+ return;
+ }
+
+ BT_DBG("idx 0x%04x", idx);
+
+ sub = bt_mesh_subnet_get(idx);
+ if (!sub) {
+ int i;
+
+ for (sub = NULL, i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ if (bt_mesh.sub[i].net_idx == BT_MESH_KEY_UNUSED) {
+ sub = &bt_mesh.sub[i];
+ break;
+ }
+ }
+
+ if (!sub) {
+ send_net_key_status(model, ctx, idx,
+ STATUS_INSUFF_RESOURCES);
+ return;
+ }
+ }
+
+ /* Check for already existing subnet */
+ if (sub->net_idx == idx) {
+ u8_t status;
+
+ if (memcmp(buf->data, sub->keys[0].net, 16)) {
+ status = STATUS_IDX_ALREADY_STORED;
+ } else {
+ status = STATUS_SUCCESS;
+ }
+
+ send_net_key_status(model, ctx, idx, status);
+ return;
+ }
+
+ err = bt_mesh_net_keys_create(&sub->keys[0], buf->data);
+ if (err) {
+ send_net_key_status(model, ctx, idx, STATUS_UNSPECIFIED);
+ return;
+ }
+
+ sub->net_idx = idx;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ BT_DBG("Storing NetKey persistently");
+ bt_mesh_store_subnet(sub);
+ }
+
+ /* Make sure we have valid beacon data to be sent */
+ bt_mesh_net_beacon_update(sub);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ sub->node_id = BT_MESH_NODE_IDENTITY_STOPPED;
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_proxy_beacon_send(sub);
+#endif
+ bt_mesh_adv_update();
+ } else {
+ sub->node_id = BT_MESH_NODE_IDENTITY_NOT_SUPPORTED;
+ }
+
+ send_net_key_status(model, ctx, idx, STATUS_SUCCESS);
+}
+
+static void net_key_update(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_subnet *sub;
+ u16_t idx;
+ int err;
+
+ idx = net_buf_simple_pull_le16(buf);
+ if (idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", idx);
+ return;
+ }
+
+ BT_DBG("idx 0x%04x", idx);
+
+ sub = bt_mesh_subnet_get(idx);
+ if (!sub) {
+ send_net_key_status(model, ctx, idx, STATUS_INVALID_NETKEY);
+ return;
+ }
+
+ /* The node shall successfully process a NetKey Update message on a
+ * valid NetKeyIndex when the NetKey value is different and the Key
+ * Refresh procedure has not been started, or when the NetKey value is
+ * the same in Phase 1. The NetKey Update message shall generate an
+ * error when the node is in Phase 2, or Phase 3.
+ */
+ switch (sub->kr_phase) {
+ case BT_MESH_KR_NORMAL:
+ if (!memcmp(buf->data, sub->keys[0].net, 16)) {
+ return;
+ }
+ break;
+ case BT_MESH_KR_PHASE_1:
+ if (!memcmp(buf->data, sub->keys[1].net, 16)) {
+ send_net_key_status(model, ctx, idx, STATUS_SUCCESS);
+ return;
+ }
+ /* fall through */
+ case BT_MESH_KR_PHASE_2:
+ case BT_MESH_KR_PHASE_3:
+ send_net_key_status(model, ctx, idx, STATUS_CANNOT_UPDATE);
+ return;
+ }
+
+ err = bt_mesh_net_keys_create(&sub->keys[1], buf->data);
+ if (!err && (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER) ||
+ IS_ENABLED(CONFIG_BT_MESH_FRIEND))) {
+ err = friend_cred_update(sub);
+ }
+
+ if (err) {
+ send_net_key_status(model, ctx, idx, STATUS_UNSPECIFIED);
+ return;
+ }
+
+ sub->kr_phase = BT_MESH_KR_PHASE_1;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ BT_DBG("Storing NetKey persistently");
+ bt_mesh_store_subnet(sub);
+ }
+
+ bt_mesh_net_beacon_update(sub);
+
+ send_net_key_status(model, ctx, idx, STATUS_SUCCESS);
+}
+
+static void hb_pub_disable(struct bt_mesh_cfg_srv *cfg)
+{
+ BT_DBG("%s", __func__);
+
+ cfg->hb_pub.dst = BT_MESH_ADDR_UNASSIGNED;
+ cfg->hb_pub.count = 0;
+ cfg->hb_pub.ttl = 0;
+ cfg->hb_pub.period = 0;
+
+ k_delayed_work_cancel(&cfg->hb_pub.timer);
+}
+
+static void net_key_del(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_subnet *sub;
+ u16_t del_idx;
+ u8_t status;
+
+ del_idx = net_buf_simple_pull_le16(buf);
+ if (del_idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", del_idx);
+ return;
+ }
+
+ BT_DBG("idx 0x%04x", del_idx);
+
+ sub = bt_mesh_subnet_get(del_idx);
+ if (!sub) {
+ /* This could be a retry of a previous attempt that had its
+ * response lost, so pretend that it was a success.
+ */
+ status = STATUS_SUCCESS;
+ goto send_status;
+ }
+
+ /* The key that the message was encrypted with cannot be removed.
+ * The NetKey List must contain a minimum of one NetKey.
+ */
+ if (ctx->net_idx == del_idx) {
+ status = STATUS_CANNOT_REMOVE;
+ goto send_status;
+ }
+
+ bt_mesh_subnet_del(sub);
+ status = STATUS_SUCCESS;
+
+send_status:
+ send_net_key_status(model, ctx, del_idx, status);
+}
+
+static void net_key_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg =
+ NET_BUF_SIMPLE(2 + 4 + IDX_LEN(CONFIG_BT_MESH_SUBNET_COUNT));
+ u16_t prev, i;
+
+ bt_mesh_model_msg_init(msg, OP_NET_KEY_LIST);
+
+ prev = BT_MESH_KEY_UNUSED;
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ if (prev == BT_MESH_KEY_UNUSED) {
+ prev = sub->net_idx;
+ continue;
+ }
+
+ key_idx_pack(msg, prev, sub->net_idx);
+ prev = BT_MESH_KEY_UNUSED;
+ }
+
+ if (prev != BT_MESH_KEY_UNUSED) {
+ net_buf_simple_add_le16(msg, prev);
+ }
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send NetKey List");
+ }
+}
+
+static void node_identity_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 4 + 4);
+ struct bt_mesh_subnet *sub;
+ u8_t node_id;
+ u16_t idx;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ idx = net_buf_simple_pull_le16(buf);
+ if (idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", idx);
+ return;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NODE_IDENTITY_STATUS);
+
+ sub = bt_mesh_subnet_get(idx);
+ if (!sub) {
+ net_buf_simple_add_u8(msg, STATUS_INVALID_NETKEY);
+ node_id = 0x00;
+ } else {
+ net_buf_simple_add_u8(msg, STATUS_SUCCESS);
+ node_id = sub->node_id;
+ }
+
+ net_buf_simple_add_le16(msg, idx);
+ net_buf_simple_add_u8(msg, node_id);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Node Identity Status");
+ }
+}
+
+static void node_identity_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 4 + 4);
+ struct bt_mesh_subnet *sub;
+ u8_t node_id;
+ u16_t idx;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ idx = net_buf_simple_pull_le16(buf);
+ if (idx > 0xfff) {
+ BT_WARN("Invalid NetKeyIndex 0x%04x", idx);
+ return;
+ }
+
+ node_id = net_buf_simple_pull_u8(buf);
+ if (node_id != 0x00 && node_id != 0x01) {
+ BT_WARN("Invalid Node ID value 0x%02x", node_id);
+ return;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_NODE_IDENTITY_STATUS);
+
+ sub = bt_mesh_subnet_get(idx);
+ if (!sub) {
+ net_buf_simple_add_u8(msg, STATUS_INVALID_NETKEY);
+ net_buf_simple_add_le16(msg, idx);
+ net_buf_simple_add_u8(msg, node_id);
+ } else {
+ net_buf_simple_add_u8(msg, STATUS_SUCCESS);
+ net_buf_simple_add_le16(msg, idx);
+
+ /* Section 4.2.11.1: "When the GATT Proxy state is set to
+ * 0x00, the Node Identity state for all subnets shall be set
+ * to 0x00 and shall not be changed."
+ */
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_ENABLED) {
+#if CONFIG_BT_MESH_NODE
+ if (node_id) {
+ bt_mesh_proxy_identity_start(sub);
+ } else {
+ bt_mesh_proxy_identity_stop(sub);
+ }
+#endif
+ bt_mesh_adv_update();
+ }
+
+ net_buf_simple_add_u8(msg, sub->node_id);
+ }
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Node Identity Status");
+ }
+}
+
+static void create_mod_app_status(struct net_buf_simple *msg,
+ struct bt_mesh_model *mod, bool vnd,
+ u16_t elem_addr, u16_t app_idx,
+ u8_t status, u8_t *mod_id)
+{
+ bt_mesh_model_msg_init(msg, OP_MOD_APP_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+ net_buf_simple_add_le16(msg, elem_addr);
+ net_buf_simple_add_le16(msg, app_idx);
+
+ if (vnd) {
+ memcpy(net_buf_simple_add(msg, 4), mod_id, 4);
+ } else {
+ memcpy(net_buf_simple_add(msg, 2), mod_id, 2);
+ }
+}
+
+static void mod_app_bind(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 9 + 4);
+ u16_t elem_addr, key_app_idx;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id, status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ key_app_idx = net_buf_simple_pull_le16(buf);
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ /* Configuration Server only allows device key based access */
+ if (model == mod) {
+ BT_ERR("Client tried to bind AppKey to Configuration Model");
+ status = STATUS_CANNOT_BIND;
+ goto send_status;
+ }
+
+ status = mod_bind(mod, key_app_idx);
+
+send_status:
+ BT_DBG("status 0x%02x", status);
+ create_mod_app_status(msg, mod, vnd, elem_addr, key_app_idx, status,
+ mod_id);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Model App Bind Status response");
+ }
+}
+
+static void mod_app_unbind(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 9 + 4);
+ u16_t elem_addr, key_app_idx;
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id, status;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ key_app_idx = net_buf_simple_pull_le16(buf);
+ mod_id = buf->data;
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_status;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_status;
+ }
+
+ status = mod_unbind(mod, key_app_idx);
+
+send_status:
+ BT_DBG("status 0x%02x", status);
+ create_mod_app_status(msg, mod, vnd, elem_addr, key_app_idx, status,
+ mod_id);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Model App Unbind Status response");
+ }
+}
+
+#define KEY_LIST_LEN (CONFIG_BT_MESH_MODEL_KEY_COUNT * 2)
+
+static void mod_app_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 9 + KEY_LIST_LEN + 4);
+ struct bt_mesh_model *mod;
+ struct bt_mesh_elem *elem;
+ u8_t *mod_id, status;
+ u16_t elem_addr;
+ bool vnd;
+
+ elem_addr = net_buf_simple_pull_le16(buf);
+ mod_id = buf->data;
+
+ BT_DBG("elem_addr 0x%04x", elem_addr);
+
+ elem = bt_mesh_elem_find(elem_addr);
+ if (!elem) {
+ mod = NULL;
+ vnd = (buf->len == 4);
+ status = STATUS_INVALID_ADDRESS;
+ goto send_list;
+ }
+
+ mod = get_model(elem, buf, &vnd);
+ if (!mod) {
+ status = STATUS_INVALID_MODEL;
+ goto send_list;
+ }
+
+ status = STATUS_SUCCESS;
+
+send_list:
+ if (vnd) {
+ bt_mesh_model_msg_init(msg, OP_VND_MOD_APP_LIST);
+ } else {
+ bt_mesh_model_msg_init(msg, OP_SIG_MOD_APP_LIST);
+ }
+
+ net_buf_simple_add_u8(msg, status);
+ net_buf_simple_add_le16(msg, elem_addr);
+
+ if (vnd) {
+ net_buf_simple_add_mem(msg, mod_id, 4);
+ } else {
+ net_buf_simple_add_mem(msg, mod_id, 2);
+ }
+
+ if (mod) {
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(mod->keys); i++) {
+ if (mod->keys[i] != BT_MESH_KEY_UNUSED) {
+ net_buf_simple_add_le16(msg, mod->keys[i]);
+ }
+ }
+ }
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Model Application List message");
+ }
+}
+
+static void node_reset(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+
+ bt_mesh_model_msg_init(msg, OP_NODE_RESET_STATUS);
+
+ /* Send the response first since we wont have any keys left to
+ * send it later.
+ */
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Node Reset Status");
+ }
+
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_reset();
+#endif
+}
+
+static void send_friend_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+
+ bt_mesh_model_msg_init(msg, OP_FRIEND_STATUS);
+ net_buf_simple_add_u8(msg, cfg->frnd);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Friend Status");
+ }
+}
+
+static void friend_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ send_friend_status(model, ctx);
+}
+
+static void friend_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+ struct bt_mesh_subnet *sub;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ if (buf->data[0] != 0x00 && buf->data[0] != 0x01) {
+ BT_WARN("Invalid Friend value 0x%02x", buf->data[0]);
+ return;
+ }
+
+ if (!cfg) {
+ BT_WARN("No Configuration Server context available");
+ goto send_status;
+ }
+
+ BT_DBG("Friend 0x%02x -> 0x%02x", cfg->frnd, buf->data[0]);
+
+ if (cfg->frnd == buf->data[0]) {
+ goto send_status;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ cfg->frnd = buf->data[0];
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_cfg();
+ }
+
+ if (cfg->frnd == BT_MESH_FRIEND_DISABLED) {
+ bt_mesh_friend_clear_net_idx(BT_MESH_KEY_ANY);
+ }
+ }
+
+ sub = bt_mesh_subnet_get(cfg->hb_pub.net_idx);
+ if ((cfg->hb_pub.feat & BT_MESH_FEAT_FRIEND) && sub) {
+ hb_send(model);
+ }
+
+send_status:
+ send_friend_status(model, ctx);
+}
+
+static void lpn_timeout_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 5 + 4);
+ struct bt_mesh_friend *frnd;
+ u16_t lpn_addr;
+ s32_t timeout;
+
+ lpn_addr = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x lpn_addr 0x%02x",
+ ctx->net_idx, ctx->app_idx, ctx->addr, lpn_addr);
+
+ if (!BT_MESH_ADDR_IS_UNICAST(lpn_addr)) {
+ BT_WARN("Invalid LPNAddress; ignoring msg");
+ return;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_LPN_TIMEOUT_STATUS);
+ net_buf_simple_add_le16(msg, lpn_addr);
+
+ if (!IS_ENABLED(CONFIG_BLUETOOTH_MESH_FRIEND)) {
+ timeout = 0;
+ goto send_rsp;
+ }
+
+ frnd = bt_mesh_friend_find(BT_MESH_KEY_ANY, lpn_addr, true, true);
+ if (!frnd) {
+ timeout = 0;
+ goto send_rsp;
+ }
+
+ timeout = k_delayed_work_remaining_get(&frnd->timer) / 100;
+
+send_rsp:
+ net_buf_simple_add_u8(msg, timeout);
+ net_buf_simple_add_u8(msg, timeout >> 8);
+ net_buf_simple_add_u8(msg, timeout >> 16);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send LPN PollTimeout Status");
+ }
+}
+
+static void send_krp_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ u16_t idx, u8_t phase, u8_t status)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 4 + 4);
+
+ bt_mesh_model_msg_init(msg, OP_KRP_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+ net_buf_simple_add_le16(msg, idx);
+ net_buf_simple_add_u8(msg, phase);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Key Refresh State Status");
+ }
+}
+
+static void krp_get(struct bt_mesh_model *model, struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_subnet *sub;
+ u16_t idx;
+
+ idx = net_buf_simple_pull_le16(buf);
+ if (idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", idx);
+ return;
+ }
+
+ BT_DBG("idx 0x%04x", idx);
+
+ sub = bt_mesh_subnet_get(idx);
+ if (!sub) {
+ send_krp_status(model, ctx, idx, 0x00, STATUS_INVALID_NETKEY);
+ } else {
+ send_krp_status(model, ctx, idx, sub->kr_phase,
+ STATUS_SUCCESS);
+ }
+}
+
+static void krp_set(struct bt_mesh_model *model, struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_subnet *sub;
+ u8_t phase;
+ u16_t idx;
+
+ idx = net_buf_simple_pull_le16(buf);
+ phase = net_buf_simple_pull_u8(buf);
+
+ if (idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", idx);
+ return;
+ }
+
+ BT_DBG("idx 0x%04x transition 0x%02x", idx, phase);
+
+ sub = bt_mesh_subnet_get(idx);
+ if (!sub) {
+ send_krp_status(model, ctx, idx, 0x00, STATUS_INVALID_NETKEY);
+ return;
+ }
+
+ BT_DBG("%u -> %u", sub->kr_phase, phase);
+
+ if (phase < BT_MESH_KR_PHASE_2 || phase > BT_MESH_KR_PHASE_3 ||
+ (sub->kr_phase == BT_MESH_KR_NORMAL &&
+ phase == BT_MESH_KR_PHASE_2)) {
+ BT_WARN("Prohibited transition %u -> %u", sub->kr_phase, phase);
+ return;
+ }
+
+ if (sub->kr_phase == BT_MESH_KR_PHASE_1 &&
+ phase == BT_MESH_KR_PHASE_2) {
+ sub->kr_phase = BT_MESH_KR_PHASE_2;
+ sub->kr_flag = 1;
+ bt_mesh_net_beacon_update(sub);
+ } else if ((sub->kr_phase == BT_MESH_KR_PHASE_1 ||
+ sub->kr_phase == BT_MESH_KR_PHASE_2) &&
+ phase == BT_MESH_KR_PHASE_3) {
+ bt_mesh_net_revoke_keys(sub);
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER) ||
+ IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ friend_cred_refresh(ctx->net_idx);
+ }
+ sub->kr_phase = BT_MESH_KR_NORMAL;
+ sub->kr_flag = 0;
+ bt_mesh_net_beacon_update(sub);
+ }
+
+ send_krp_status(model, ctx, idx, sub->kr_phase, STATUS_SUCCESS);
+}
+
+static u8_t hb_log(u16_t val)
+{
+ if (!val) {
+ return 0x00;
+ } else if (val == 0xffff) {
+ return 0xff;
+ } else {
+ return 32 - __builtin_clz(val);
+ }
+}
+
+static u8_t hb_pub_count_log(u16_t val)
+{
+ if (!val) {
+ return 0x00;
+ } else if (val == 0x01) {
+ return 0x01;
+ } else if (val == 0xffff) {
+ return 0xff;
+ } else {
+ return 32 - __builtin_clz(val - 1) + 1;
+ }
+}
+
+static u16_t hb_pwr2(u8_t val, u8_t sub)
+{
+ if (!val) {
+ return 0x0000;
+ } else if (val == 0xff || val == 0x11) {
+ return 0xffff;
+ } else {
+ return (1 << (val - sub));
+ }
+}
+
+struct hb_pub_param {
+ u16_t dst;
+ u8_t count_log;
+ u8_t period_log;
+ u8_t ttl;
+ u16_t feat;
+ u16_t net_idx;
+} __packed;
+
+static void hb_pub_send_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, u8_t status,
+ struct hb_pub_param *orig_msg)
+{
+ /* Needed size: opcode (1 byte) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(1 + 10 + 4);
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+
+ BT_DBG("src 0x%04x status 0x%02x", ctx->addr, status);
+
+ bt_mesh_model_msg_init(msg, OP_HEARTBEAT_PUB_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+
+ if (orig_msg) {
+ memcpy(net_buf_simple_add(msg, sizeof(*orig_msg)), orig_msg,
+ sizeof(*orig_msg));
+ goto send;
+ }
+
+ net_buf_simple_add_le16(msg, cfg->hb_pub.dst);
+ net_buf_simple_add_u8(msg, hb_pub_count_log(cfg->hb_pub.count));
+ net_buf_simple_add_u8(msg, cfg->hb_pub.period);
+ net_buf_simple_add_u8(msg, cfg->hb_pub.ttl);
+ net_buf_simple_add_le16(msg, cfg->hb_pub.feat);
+ net_buf_simple_add_le16(msg, cfg->hb_pub.net_idx);
+
+send:
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Heartbeat Publication Status");
+ }
+}
+
+static void heartbeat_pub_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("src 0x%04x", ctx->addr);
+
+ hb_pub_send_status(model, ctx, STATUS_SUCCESS, NULL);
+}
+
+static void heartbeat_pub_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct hb_pub_param *param = (void *)buf->data;
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+ u16_t dst, feat, idx;
+ u8_t status;
+
+ BT_DBG("src 0x%04x", ctx->addr);
+
+ dst = sys_le16_to_cpu(param->dst);
+ /* All address types are valid, except for virtual ones */
+ if (BT_MESH_ADDR_IS_VIRTUAL(dst)) {
+ status = STATUS_INVALID_ADDRESS;
+ goto failed;
+ }
+
+ if (param->count_log > 0x11 && param->count_log != 0xff) {
+ status = STATUS_CANNOT_SET;
+ goto failed;
+ }
+
+ if (param->period_log > 0x10) {
+ status = STATUS_CANNOT_SET;
+ goto failed;
+ }
+
+ if (param->ttl > BT_MESH_TTL_MAX && param->ttl != BT_MESH_TTL_DEFAULT) {
+ BT_ERR("Invalid TTL value 0x%02x", param->ttl);
+ return;
+ }
+
+ feat = sys_le16_to_cpu(param->feat);
+
+ idx = sys_le16_to_cpu(param->net_idx);
+ if (idx > 0xfff) {
+ BT_ERR("Invalid NetKeyIndex 0x%04x", idx);
+ return;
+ }
+
+ if (!bt_mesh_subnet_get(idx)) {
+ status = STATUS_INVALID_NETKEY;
+ goto failed;
+ }
+
+ cfg->hb_pub.dst = dst;
+ cfg->hb_pub.period = param->period_log;
+ cfg->hb_pub.feat = feat;
+ cfg->hb_pub.net_idx = idx;
+
+ if (dst == BT_MESH_ADDR_UNASSIGNED) {
+ hb_pub_disable(cfg);
+ } else {
+ /* 2^(n-1) */
+ cfg->hb_pub.count = hb_pwr2(param->count_log, 1);
+ cfg->hb_pub.ttl = param->ttl;
+
+ BT_DBG("period %u ms", hb_pwr2(param->period_log, 1) * 1000);
+
+ /* Note: Send heartbeat message here will cause wrong heartbeat status message */
+#if 0
+ /* The first Heartbeat message shall be published as soon
+ * as possible after the Heartbeat Publication Period state
+ * has been configured for periodic publishing.
+ */
+ if (param->period_log && param->count_log) {
+ k_work_submit(&cfg->hb_pub.timer.work);
+ } else {
+ k_delayed_work_cancel(&cfg->hb_pub.timer);
+ }
+#endif
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_hb_pub();
+ }
+
+ hb_pub_send_status(model, ctx, STATUS_SUCCESS, NULL);
+
+ /* The first Heartbeat message shall be published as soon
+ * as possible after the Heartbeat Publication Period state
+ * has been configured for periodic publishing.
+ */
+ if (dst != BT_MESH_ADDR_UNASSIGNED) {
+ if (param->period_log && param->count_log) {
+ k_work_submit(&cfg->hb_pub.timer.work);
+ } else {
+ k_delayed_work_cancel(&cfg->hb_pub.timer);
+ }
+ }
+
+ return;
+
+failed:
+ hb_pub_send_status(model, ctx, status, param);
+}
+
+static void hb_sub_send_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx, u8_t status)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 9 + 4);
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+ u16_t period;
+ s64_t uptime;
+
+ BT_DBG("src 0x%04x status 0x%02x", ctx->addr, status);
+
+ uptime = k_uptime_get();
+ if (uptime > cfg->hb_sub.expiry) {
+ period = 0;
+ } else {
+ period = (cfg->hb_sub.expiry - uptime) / 1000;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEARTBEAT_SUB_STATUS);
+
+ net_buf_simple_add_u8(msg, status);
+
+ net_buf_simple_add_le16(msg, cfg->hb_sub.src);
+ net_buf_simple_add_le16(msg, cfg->hb_sub.dst);
+
+ if (cfg->hb_sub.src == BT_MESH_ADDR_UNASSIGNED ||
+ cfg->hb_sub.dst == BT_MESH_ADDR_UNASSIGNED) {
+ memset(net_buf_simple_add(msg, 4), 0, 4);
+ } else {
+ net_buf_simple_add_u8(msg, hb_log(period));
+ net_buf_simple_add_u8(msg, hb_log(cfg->hb_sub.count));
+ net_buf_simple_add_u8(msg, cfg->hb_sub.min_hops);
+ net_buf_simple_add_u8(msg, cfg->hb_sub.max_hops);
+ }
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Heartbeat Subscription Status");
+ }
+}
+
+static void heartbeat_sub_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("src 0x%04x", ctx->addr);
+
+ hb_sub_send_status(model, ctx, STATUS_SUCCESS);
+}
+
+static void heartbeat_sub_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+ u16_t sub_src, sub_dst;
+ u8_t sub_period;
+ s32_t period_ms;
+
+ BT_DBG("src 0x%04x", ctx->addr);
+
+ sub_src = net_buf_simple_pull_le16(buf);
+ sub_dst = net_buf_simple_pull_le16(buf);
+ sub_period = net_buf_simple_pull_u8(buf);
+
+ BT_DBG("sub_src 0x%04x sub_dst 0x%04x period 0x%02x",
+ sub_src, sub_dst, sub_period);
+
+ if (sub_src != BT_MESH_ADDR_UNASSIGNED &&
+ !BT_MESH_ADDR_IS_UNICAST(sub_src)) {
+ BT_WARN("Prohibited source address");
+ return;
+ }
+
+ if (BT_MESH_ADDR_IS_VIRTUAL(sub_dst) || BT_MESH_ADDR_IS_RFU(sub_dst) ||
+ (BT_MESH_ADDR_IS_UNICAST(sub_dst) &&
+ sub_dst != bt_mesh_primary_addr())) {
+ BT_WARN("Prohibited destination address");
+ return;
+ }
+
+ if (sub_period > 0x11) {
+ BT_WARN("Prohibited subscription period 0x%02x", sub_period);
+ return;
+ }
+
+ if (sub_src == BT_MESH_ADDR_UNASSIGNED ||
+ sub_dst == BT_MESH_ADDR_UNASSIGNED ||
+ sub_period == 0x00) {
+ /* Setting the same addresses with zero period should retain
+ * the addresses according to MESH/NODE/CFG/HBS/BV-02-C.
+ */
+ if (cfg->hb_sub.src != sub_src || cfg->hb_sub.dst != sub_dst) {
+ cfg->hb_sub.src = BT_MESH_ADDR_UNASSIGNED;
+ cfg->hb_sub.dst = BT_MESH_ADDR_UNASSIGNED;
+ }
+
+ period_ms = 0;
+ } else {
+ cfg->hb_sub.src = sub_src;
+ cfg->hb_sub.dst = sub_dst;
+ cfg->hb_sub.min_hops = BT_MESH_TTL_MAX;
+ cfg->hb_sub.max_hops = 0;
+ cfg->hb_sub.count = 0;
+ period_ms = hb_pwr2(sub_period, 1) * 1000;
+ }
+
+ /* Let the transport layer know it needs to handle this address */
+ bt_mesh_set_hb_sub_dst(cfg->hb_sub.dst);
+
+ BT_DBG("period_ms %u", period_ms);
+
+ if (period_ms) {
+ cfg->hb_sub.expiry = k_uptime_get() + period_ms;
+ } else {
+ cfg->hb_sub.expiry = 0;
+ }
+
+ hb_sub_send_status(model, ctx, STATUS_SUCCESS);
+}
+
+const struct bt_mesh_model_op bt_mesh_cfg_srv_op[] = {
+ { OP_DEV_COMP_DATA_GET, 1, dev_comp_data_get },
+ { OP_APP_KEY_ADD, 19, app_key_add },
+ { OP_APP_KEY_UPDATE, 19, app_key_update },
+ { OP_APP_KEY_DEL, 3, app_key_del },
+ { OP_APP_KEY_GET, 2, app_key_get },
+ { OP_BEACON_GET, 0, beacon_get },
+ { OP_BEACON_SET, 1, beacon_set },
+ { OP_DEFAULT_TTL_GET, 0, default_ttl_get },
+ { OP_DEFAULT_TTL_SET, 1, default_ttl_set },
+ { OP_GATT_PROXY_GET, 0, gatt_proxy_get },
+ { OP_GATT_PROXY_SET, 1, gatt_proxy_set },
+ { OP_NET_TRANSMIT_GET, 0, net_transmit_get },
+ { OP_NET_TRANSMIT_SET, 1, net_transmit_set },
+ { OP_RELAY_GET, 0, relay_get },
+ { OP_RELAY_SET, 2, relay_set },
+ { OP_MOD_PUB_GET, 4, mod_pub_get },
+ { OP_MOD_PUB_SET, 11, mod_pub_set },
+ { OP_MOD_PUB_VA_SET, 24, mod_pub_va_set },
+ { OP_MOD_SUB_ADD, 6, mod_sub_add },
+ { OP_MOD_SUB_VA_ADD, 20, mod_sub_va_add },
+ { OP_MOD_SUB_DEL, 6, mod_sub_del },
+ { OP_MOD_SUB_VA_DEL, 20, mod_sub_va_del },
+ { OP_MOD_SUB_OVERWRITE, 6, mod_sub_overwrite },
+ { OP_MOD_SUB_VA_OVERWRITE, 20, mod_sub_va_overwrite },
+ { OP_MOD_SUB_DEL_ALL, 4, mod_sub_del_all },
+ { OP_MOD_SUB_GET, 4, mod_sub_get },
+ { OP_MOD_SUB_GET_VND, 6, mod_sub_get_vnd },
+ { OP_NET_KEY_ADD, 18, net_key_add },
+ { OP_NET_KEY_UPDATE, 18, net_key_update },
+ { OP_NET_KEY_DEL, 2, net_key_del },
+ { OP_NET_KEY_GET, 0, net_key_get },
+ { OP_NODE_IDENTITY_GET, 2, node_identity_get },
+ { OP_NODE_IDENTITY_SET, 3, node_identity_set },
+ { OP_MOD_APP_BIND, 6, mod_app_bind },
+ { OP_MOD_APP_UNBIND, 6, mod_app_unbind },
+ { OP_SIG_MOD_APP_GET, 4, mod_app_get },
+ { OP_VND_MOD_APP_GET, 6, mod_app_get },
+ { OP_NODE_RESET, 0, node_reset },
+ { OP_FRIEND_GET, 0, friend_get },
+ { OP_FRIEND_SET, 1, friend_set },
+ { OP_LPN_TIMEOUT_GET, 2, lpn_timeout_get },
+ { OP_KRP_GET, 2, krp_get },
+ { OP_KRP_SET, 3, krp_set },
+ { OP_HEARTBEAT_PUB_GET, 0, heartbeat_pub_get },
+ { OP_HEARTBEAT_PUB_SET, 9, heartbeat_pub_set },
+ { OP_HEARTBEAT_SUB_GET, 0, heartbeat_sub_get },
+ { OP_HEARTBEAT_SUB_SET, 5, heartbeat_sub_set },
+ BT_MESH_MODEL_OP_END,
+};
+
+static void hb_publish(struct k_work *work)
+{
+ struct bt_mesh_cfg_srv *cfg = CONTAINER_OF(work,
+ struct bt_mesh_cfg_srv,
+ hb_pub.timer.work);
+ struct bt_mesh_model *model = cfg->model;
+ struct bt_mesh_subnet *sub;
+ u16_t period_ms;
+
+ BT_DBG("hb_pub.count: %u", cfg->hb_pub.count);
+
+ sub = bt_mesh_subnet_get(cfg->hb_pub.net_idx);
+ if (!sub) {
+ BT_ERR("No matching subnet for idx 0x%02x",
+ cfg->hb_pub.net_idx);
+ cfg->hb_pub.dst = BT_MESH_ADDR_UNASSIGNED;
+ return;
+ }
+
+ if (cfg->hb_pub.count == 0) {
+ return;
+ }
+
+ period_ms = hb_pwr2(cfg->hb_pub.period, 1) * 1000;
+ if (period_ms && cfg->hb_pub.count > 1) {
+ k_delayed_work_submit(&cfg->hb_pub.timer, period_ms);
+ }
+
+ hb_send(model);
+
+ if (cfg->hb_pub.count != 0xffff) {
+ cfg->hb_pub.count--;
+ }
+}
+
+static bool conf_is_valid(struct bt_mesh_cfg_srv *cfg)
+{
+ if (cfg->relay > 0x02) {
+ return false;
+ }
+
+ if (cfg->beacon > 0x01) {
+ return false;
+ }
+
+ if (cfg->default_ttl > BT_MESH_TTL_MAX) {
+ return false;
+ }
+
+ return true;
+}
+
+int bt_mesh_cfg_srv_init(struct bt_mesh_model *model, bool primary)
+{
+ struct bt_mesh_cfg_srv *cfg = model->user_data;
+
+ if (!cfg) {
+ BT_ERR("No Configuration Server context provided");
+ return -EINVAL;
+ }
+
+ BT_DBG("%s, net_transmit = %d, relay = %d, relay_retransmit = %d, beacon = %d, gatt_proxy = %d, frnd = %d, default_ttl = %d",
+ __func__, cfg->net_transmit, cfg->relay, cfg->relay_retransmit, cfg->beacon, cfg->gatt_proxy, cfg->frnd, cfg->default_ttl);
+
+ if (!conf_is_valid(cfg)) {
+ BT_ERR("Invalid values in configuration");
+ return -EINVAL;
+ }
+
+ /* Configuration Model security is device-key based */
+ model->keys[0] = BT_MESH_KEY_DEV;
+
+ if (!IS_ENABLED(CONFIG_BT_MESH_RELAY)) {
+ cfg->relay = BT_MESH_RELAY_NOT_SUPPORTED;
+ }
+
+ if (!IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ cfg->frnd = BT_MESH_FRIEND_NOT_SUPPORTED;
+ }
+
+ if (!IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ cfg->gatt_proxy = BT_MESH_GATT_PROXY_NOT_SUPPORTED;
+ }
+
+ k_delayed_work_init(&cfg->hb_pub.timer, hb_publish);
+ cfg->hb_sub.expiry = 0;
+
+ cfg->model = model;
+
+ conf = cfg;
+
+ return 0;
+}
+
+static void mod_reset(struct bt_mesh_model *mod, struct bt_mesh_elem *elem,
+ bool vnd, bool primary, void *user_data)
+{
+ /* Clear model state that isn't otherwise cleared. E.g. AppKey
+ * binding and model publication is cleared as a consequence
+ * of removing all app keys, however model subscription clearing
+ * must be taken care of here.
+ */
+
+ mod_sub_list_clear(mod);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_mod_sub(mod);
+ }
+}
+
+void bt_mesh_cfg_reset(void)
+{
+ struct bt_mesh_cfg_srv *cfg = conf;
+ int i;
+
+ if (!cfg) {
+ return;
+ }
+
+ bt_mesh_set_hb_sub_dst(BT_MESH_ADDR_UNASSIGNED);
+
+ cfg->hb_sub.src = BT_MESH_ADDR_UNASSIGNED;
+ cfg->hb_sub.dst = BT_MESH_ADDR_UNASSIGNED;
+ cfg->hb_sub.expiry = 0;
+
+ /* Delete all net keys, which also takes care of all app keys which
+ * are associated with each net key.
+ */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx != BT_MESH_KEY_UNUSED) {
+ bt_mesh_subnet_del(sub);
+ }
+ }
+
+ bt_mesh_model_foreach(mod_reset, NULL);
+
+ memset(labels, 0, sizeof(labels));
+}
+
+void bt_mesh_heartbeat(u16_t src, u16_t dst, u8_t hops, u16_t feat)
+{
+ struct bt_mesh_cfg_srv *cfg = conf;
+
+ if (!cfg) {
+ BT_WARN("No configuaration server context available");
+ return;
+ }
+
+ if (src != cfg->hb_sub.src || dst != cfg->hb_sub.dst) {
+ BT_WARN("No subscription for received heartbeat");
+ return;
+ }
+
+ if (k_uptime_get() > cfg->hb_sub.expiry) {
+ BT_WARN("Heartbeat subscription period expired");
+ return;
+ }
+
+ cfg->hb_sub.min_hops = min(cfg->hb_sub.min_hops, hops);
+ cfg->hb_sub.max_hops = max(cfg->hb_sub.max_hops, hops);
+
+ if (cfg->hb_sub.count < 0xffff) {
+ cfg->hb_sub.count++;
+ }
+
+ BT_DBG("src 0x%04x dst 0x%04x hops %u min %u max %u count %u", src,
+ dst, hops, cfg->hb_sub.min_hops, cfg->hb_sub.max_hops,
+ cfg->hb_sub.count);
+
+ if (cfg->hb_sub.func) {
+ cfg->hb_sub.func(hops, feat);
+ }
+}
+
+u8_t bt_mesh_net_transmit_get(void)
+{
+ if (conf) {
+ return conf->net_transmit;
+ }
+
+ return 0;
+}
+
+u8_t bt_mesh_relay_get(void)
+{
+ if (conf) {
+ return conf->relay;
+ }
+
+ return BT_MESH_RELAY_NOT_SUPPORTED;
+}
+
+u8_t bt_mesh_friend_get(void)
+{
+ if (conf) {
+ BT_DBG("conf %p conf->frnd 0x%02x", conf, conf->frnd);
+ return conf->frnd;
+ }
+
+ return BT_MESH_FRIEND_NOT_SUPPORTED;
+}
+
+u8_t bt_mesh_relay_retransmit_get(void)
+{
+ if (conf) {
+ return conf->relay_retransmit;
+ }
+
+ return 0;
+}
+
+u8_t bt_mesh_beacon_get(void)
+{
+ if (conf) {
+ return conf->beacon;
+ }
+
+ return BT_MESH_BEACON_DISABLED;
+}
+
+u8_t bt_mesh_gatt_proxy_get(void)
+{
+ if (conf) {
+ return conf->gatt_proxy;
+ }
+
+ return BT_MESH_GATT_PROXY_NOT_SUPPORTED;
+}
+
+u8_t bt_mesh_default_ttl_get(void)
+{
+ if (conf) {
+ return conf->default_ttl;
+ }
+
+ return DEFAULT_TTL;
+}
+
+u8_t *bt_mesh_label_uuid_get(u16_t addr)
+{
+ int i;
+
+ BT_DBG("addr 0x%04x", addr);
+
+ for (i = 0; i < ARRAY_SIZE(labels); i++) {
+ if (labels[i].addr == addr) {
+ BT_DBG("Found Label UUID for 0x%04x: %s", addr,
+ bt_hex(labels[i].uuid, 16));
+ return labels[i].uuid;
+ }
+ }
+
+ BT_WARN("No matching Label UUID for 0x%04x", addr);
+
+ return NULL;
+}
+
+struct bt_mesh_hb_pub *bt_mesh_hb_pub_get(void)
+{
+ if (!conf) {
+ return NULL;
+ }
+
+ return &conf->hb_pub;
+}
+
+struct bt_mesh_cfg_srv *bt_mesh_cfg_get(void)
+{
+ return conf;
+}
+
+void bt_mesh_subnet_del(struct bt_mesh_subnet *sub)
+{
+ int i;
+
+ if (conf && conf->hb_pub.net_idx == sub->net_idx) {
+ hb_pub_disable(conf);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_hb_pub();
+ }
+ }
+
+ /* Delete any app keys bound to this NetKey index */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ struct bt_mesh_app_key *key = &bt_mesh.app_keys[i];
+
+ if (key->net_idx == sub->net_idx) {
+ bt_mesh_app_key_del(key);
+ }
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ bt_mesh_friend_clear_net_idx(sub->net_idx);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_clear_subnet(sub);
+ }
+
+ memset(sub, 0, sizeof(*sub));
+ sub->net_idx = BT_MESH_KEY_UNUSED;
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/crypto.c b/components/bt/ble_mesh/mesh_core/crypto.c
new file mode 100644
index 0000000000..a0b1cb2493
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/crypto.c
@@ -0,0 +1,880 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+#include "mesh_types.h"
+#include "mesh_util.h"
+#include "sdkconfig.h"
+#include "mesh_buf.h"
+
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_CRYPTO)
+#include "mesh_trace.h"
+
+#include "mesh.h"
+#include "crypto.h"
+#include "mesh_bearer_adapt.h"
+#include "mesh_aes_encrypt.h"
+
+#define NET_MIC_LEN(pdu) (((pdu)[1] & 0x80) ? 8 : 4)
+#define APP_MIC_LEN(aszmic) ((aszmic) ? 8 : 4)
+
+int bt_mesh_aes_cmac(const u8_t key[16], struct bt_mesh_sg *sg,
+ size_t sg_len, u8_t mac[16])
+{
+ struct tc_aes_key_sched_struct sched;
+ struct tc_cmac_struct state;
+
+ if (tc_cmac_setup(&state, key, &sched) == TC_CRYPTO_FAIL) {
+ return -EIO;
+ }
+
+ for (; sg_len; sg_len--, sg++) {
+ if (tc_cmac_update(&state, sg->data,
+ sg->len) == TC_CRYPTO_FAIL) {
+ return -EIO;
+ }
+ }
+
+ if (tc_cmac_final(mac, &state) == TC_CRYPTO_FAIL) {
+ return -EIO;
+ }
+ return 0;
+}
+
+int bt_mesh_k1(const u8_t *ikm, size_t ikm_len, const u8_t salt[16],
+ const char *info, u8_t okm[16])
+{
+ int err;
+
+ err = bt_mesh_aes_cmac_one(salt, ikm, ikm_len, okm);
+ if (err < 0) {
+ return err;
+ }
+
+ return bt_mesh_aes_cmac_one(okm, info, strlen(info), okm);
+}
+
+int bt_mesh_k2(const u8_t n[16], const u8_t *p, size_t p_len,
+ u8_t net_id[1], u8_t enc_key[16], u8_t priv_key[16])
+{
+ struct bt_mesh_sg sg[3];
+ u8_t salt[16];
+ u8_t out[16];
+ u8_t t[16];
+ u8_t pad;
+ int err;
+
+ BT_DBG("n %s", bt_hex(n, 16));
+ BT_DBG("p %s", bt_hex(p, p_len));
+
+ err = bt_mesh_s1("smk2", salt);
+ if (err) {
+ return err;
+ }
+
+ err = bt_mesh_aes_cmac_one(salt, n, 16, t);
+ if (err) {
+ return err;
+ }
+
+ pad = 0x01;
+
+ sg[0].data = NULL;
+ sg[0].len = 0;
+ sg[1].data = p;
+ sg[1].len = p_len;
+ sg[2].data = &pad;
+ sg[2].len = sizeof(pad);
+
+ err = bt_mesh_aes_cmac(t, sg, ARRAY_SIZE(sg), out);
+ if (err) {
+ return err;
+ }
+
+ net_id[0] = out[15] & 0x7f;
+
+ sg[0].data = out;
+ sg[0].len = sizeof(out);
+ pad = 0x02;
+
+ err = bt_mesh_aes_cmac(t, sg, ARRAY_SIZE(sg), out);
+ if (err) {
+ return err;
+ }
+
+ memcpy(enc_key, out, 16);
+
+ pad = 0x03;
+
+ err = bt_mesh_aes_cmac(t, sg, ARRAY_SIZE(sg), out);
+ if (err) {
+ return err;
+ }
+
+ memcpy(priv_key, out, 16);
+
+ BT_DBG("NID 0x%02x enc_key %s", net_id[0], bt_hex(enc_key, 16));
+ BT_DBG("priv_key %s", bt_hex(priv_key, 16));
+
+ return 0;
+}
+
+int bt_mesh_k3(const u8_t n[16], u8_t out[8])
+{
+ u8_t id64[] = { 'i', 'd', '6', '4', 0x01 };
+ u8_t tmp[16];
+ u8_t t[16];
+ int err;
+
+ err = bt_mesh_s1("smk3", tmp);
+ if (err) {
+ return err;
+ }
+
+ err = bt_mesh_aes_cmac_one(tmp, n, 16, t);
+ if (err) {
+ return err;
+ }
+
+ err = bt_mesh_aes_cmac_one(t, id64, sizeof(id64), tmp);
+ if (err) {
+ return err;
+ }
+
+ memcpy(out, tmp + 8, 8);
+
+ return 0;
+}
+
+int bt_mesh_k4(const u8_t n[16], u8_t out[1])
+{
+ u8_t id6[] = { 'i', 'd', '6', 0x01 };
+ u8_t tmp[16];
+ u8_t t[16];
+ int err;
+
+ err = bt_mesh_s1("smk4", tmp);
+ if (err) {
+ return err;
+ }
+
+ err = bt_mesh_aes_cmac_one(tmp, n, 16, t);
+ if (err) {
+ return err;
+ }
+
+ err = bt_mesh_aes_cmac_one(t, id6, sizeof(id6), tmp);
+ if (err) {
+ return err;
+ }
+
+ out[0] = tmp[15] & BIT_MASK(6);
+
+ return 0;
+}
+
+int bt_mesh_id128(const u8_t n[16], const char *s, u8_t out[16])
+{
+ const char *id128 = "id128\x01";
+ u8_t salt[16];
+ int err;
+
+ err = bt_mesh_s1(s, salt);
+ if (err) {
+ return err;
+ }
+
+ return bt_mesh_k1(n, 16, salt, id128, out);
+}
+
+static int bt_mesh_ccm_decrypt(const u8_t key[16], u8_t nonce[13],
+ const u8_t *enc_msg, size_t msg_len,
+ const u8_t *aad, size_t aad_len,
+ u8_t *out_msg, size_t mic_size)
+{
+ u8_t msg[16], pmsg[16], cmic[16], cmsg[16], Xn[16], mic[16];
+ u16_t last_blk, blk_cnt;
+ size_t i, j;
+ int err;
+
+ if (msg_len < 1 || aad_len >= 0xff00) {
+ return -EINVAL;
+ }
+
+ /* C_mic = e(AppKey, 0x01 || nonce || 0x0000) */
+ pmsg[0] = 0x01;
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(0x0000, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, cmic);
+ if (err) {
+ return err;
+ }
+
+ /* X_0 = e(AppKey, 0x09 || nonce || length) */
+ if (mic_size == sizeof(u64_t)) {
+ pmsg[0] = 0x19 | (aad_len ? 0x40 : 0x00);
+ } else {
+ pmsg[0] = 0x09 | (aad_len ? 0x40 : 0x00);
+ }
+
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(msg_len, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+
+ /* If AAD is being used to authenticate, include it here */
+ if (aad_len) {
+ sys_put_be16(aad_len, pmsg);
+
+ for (i = 0; i < sizeof(u16_t); i++) {
+ pmsg[i] = Xn[i] ^ pmsg[i];
+ }
+
+ j = 0;
+ aad_len += sizeof(u16_t);
+ while (aad_len > 16) {
+ do {
+ pmsg[i] = Xn[i] ^ aad[j];
+ i++, j++;
+ } while (i < 16);
+
+ aad_len -= 16;
+ i = 0;
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+ }
+
+ for (i = 0; i < aad_len; i++, j++) {
+ pmsg[i] = Xn[i] ^ aad[j];
+ }
+
+ for (i = aad_len; i < 16; i++) {
+ pmsg[i] = Xn[i];
+ }
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+ }
+
+ last_blk = msg_len % 16;
+ blk_cnt = (msg_len + 15) / 16;
+ if (!last_blk) {
+ last_blk = 16;
+ }
+
+ for (j = 0; j < blk_cnt; j++) {
+ if (j + 1 == blk_cnt) {
+ /* C_1 = e(AppKey, 0x01 || nonce || 0x0001) */
+ pmsg[0] = 0x01;
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(j + 1, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, cmsg);
+ if (err) {
+ return err;
+ }
+
+ /* Encrypted = Payload[0-15] ^ C_1 */
+ for (i = 0; i < last_blk; i++) {
+ msg[i] = enc_msg[(j * 16) + i] ^ cmsg[i];
+ }
+
+ memcpy(out_msg + (j * 16), msg, last_blk);
+
+ /* X_1 = e(AppKey, X_0 ^ Payload[0-15]) */
+ for (i = 0; i < last_blk; i++) {
+ pmsg[i] = Xn[i] ^ msg[i];
+ }
+
+ for (i = last_blk; i < 16; i++) {
+ pmsg[i] = Xn[i] ^ 0x00;
+ }
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+
+ /* MIC = C_mic ^ X_1 */
+ for (i = 0; i < sizeof(mic); i++) {
+ mic[i] = cmic[i] ^ Xn[i];
+ }
+ } else {
+ /* C_1 = e(AppKey, 0x01 || nonce || 0x0001) */
+ pmsg[0] = 0x01;
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(j + 1, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, cmsg);
+ if (err) {
+ return err;
+ }
+
+ /* Encrypted = Payload[0-15] ^ C_1 */
+ for (i = 0; i < 16; i++) {
+ msg[i] = enc_msg[(j * 16) + i] ^ cmsg[i];
+ }
+
+ memcpy(out_msg + (j * 16), msg, 16);
+
+ /* X_1 = e(AppKey, X_0 ^ Payload[0-15]) */
+ for (i = 0; i < 16; i++) {
+ pmsg[i] = Xn[i] ^ msg[i];
+ }
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+ }
+ }
+
+ if (memcmp(mic, enc_msg + msg_len, mic_size)) {
+ return -EBADMSG;
+ }
+
+ return 0;
+}
+
+static int bt_mesh_ccm_encrypt(const u8_t key[16], u8_t nonce[13],
+ const u8_t *msg, size_t msg_len,
+ const u8_t *aad, size_t aad_len,
+ u8_t *out_msg, size_t mic_size)
+{
+ u8_t pmsg[16], cmic[16], cmsg[16], mic[16], Xn[16];
+ u16_t blk_cnt, last_blk;
+ size_t i, j;
+ int err;
+
+ BT_DBG("key %s", bt_hex(key, 16));
+ BT_DBG("nonce %s", bt_hex(nonce, 13));
+ BT_DBG("msg (len %u) %s", msg_len, bt_hex(msg, msg_len));
+ BT_DBG("aad_len %u mic_size %u", aad_len, mic_size);
+
+ /* Unsupported AAD size */
+ if (aad_len >= 0xff00) {
+ return -EINVAL;
+ }
+
+ /* C_mic = e(AppKey, 0x01 || nonce || 0x0000) */
+ pmsg[0] = 0x01;
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(0x0000, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, cmic);
+ if (err) {
+ return err;
+ }
+
+ /* X_0 = e(AppKey, 0x09 || nonce || length) */
+ if (mic_size == sizeof(u64_t)) {
+ pmsg[0] = 0x19 | (aad_len ? 0x40 : 0x00);
+ } else {
+ pmsg[0] = 0x09 | (aad_len ? 0x40 : 0x00);
+ }
+
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(msg_len, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+
+ /* If AAD is being used to authenticate, include it here */
+ if (aad_len) {
+ sys_put_be16(aad_len, pmsg);
+
+ for (i = 0; i < sizeof(u16_t); i++) {
+ pmsg[i] = Xn[i] ^ pmsg[i];
+ }
+
+ j = 0;
+ aad_len += sizeof(u16_t);
+ while (aad_len > 16) {
+ do {
+ pmsg[i] = Xn[i] ^ aad[j];
+ i++, j++;
+ } while (i < 16);
+
+ aad_len -= 16;
+ i = 0;
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+ }
+
+ for (i = 0; i < aad_len; i++, j++) {
+ pmsg[i] = Xn[i] ^ aad[j];
+ }
+
+ for (i = aad_len; i < 16; i++) {
+ pmsg[i] = Xn[i];
+ }
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+ }
+
+ last_blk = msg_len % 16;
+ blk_cnt = (msg_len + 15) / 16;
+ if (!last_blk) {
+ last_blk = 16;
+ }
+
+ for (j = 0; j < blk_cnt; j++) {
+ if (j + 1 == blk_cnt) {
+ /* X_1 = e(AppKey, X_0 ^ Payload[0-15]) */
+ for (i = 0; i < last_blk; i++) {
+ pmsg[i] = Xn[i] ^ msg[(j * 16) + i];
+ }
+ for (i = last_blk; i < 16; i++) {
+ pmsg[i] = Xn[i] ^ 0x00;
+ }
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+
+ /* MIC = C_mic ^ X_1 */
+ for (i = 0; i < sizeof(mic); i++) {
+ mic[i] = cmic[i] ^ Xn[i];
+ }
+
+ /* C_1 = e(AppKey, 0x01 || nonce || 0x0001) */
+ pmsg[0] = 0x01;
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(j + 1, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, cmsg);
+ if (err) {
+ return err;
+ }
+
+ /* Encrypted = Payload[0-15] ^ C_1 */
+ for (i = 0; i < last_blk; i++) {
+ out_msg[(j * 16) + i] =
+ msg[(j * 16) + i] ^ cmsg[i];
+ }
+ } else {
+ /* X_1 = e(AppKey, X_0 ^ Payload[0-15]) */
+ for (i = 0; i < 16; i++) {
+ pmsg[i] = Xn[i] ^ msg[(j * 16) + i];
+ }
+
+ err = bt_encrypt_be(key, pmsg, Xn);
+ if (err) {
+ return err;
+ }
+
+ /* C_1 = e(AppKey, 0x01 || nonce || 0x0001) */
+ pmsg[0] = 0x01;
+ memcpy(pmsg + 1, nonce, 13);
+ sys_put_be16(j + 1, pmsg + 14);
+
+ err = bt_encrypt_be(key, pmsg, cmsg);
+ if (err) {
+ return err;
+ }
+
+ /* Encrypted = Payload[0-15] ^ C_N */
+ for (i = 0; i < 16; i++) {
+ out_msg[(j * 16) + i] =
+ msg[(j * 16) + i] ^ cmsg[i];
+ }
+
+ }
+ }
+
+ memcpy(out_msg + msg_len, mic, mic_size);
+
+ return 0;
+}
+
+#if defined(CONFIG_BT_MESH_PROXY)
+static void create_proxy_nonce(u8_t nonce[13], const u8_t *pdu,
+ u32_t iv_index)
+{
+ /* Nonce Type */
+ nonce[0] = 0x03;
+
+ /* Pad */
+ nonce[1] = 0x00;
+
+ /* Sequence Number */
+ nonce[2] = pdu[2];
+ nonce[3] = pdu[3];
+ nonce[4] = pdu[4];
+
+ /* Source Address */
+ nonce[5] = pdu[5];
+ nonce[6] = pdu[6];
+
+ /* Pad */
+ nonce[7] = 0;
+ nonce[8] = 0;
+
+ /* IV Index */
+ sys_put_be32(iv_index, &nonce[9]);
+}
+#endif /* PROXY */
+
+static void create_net_nonce(u8_t nonce[13], const u8_t *pdu,
+ u32_t iv_index)
+{
+ /* Nonce Type */
+ nonce[0] = 0x00;
+
+ /* FRND + TTL */
+ nonce[1] = pdu[1];
+
+ /* Sequence Number */
+ nonce[2] = pdu[2];
+ nonce[3] = pdu[3];
+ nonce[4] = pdu[4];
+
+ /* Source Address */
+ nonce[5] = pdu[5];
+ nonce[6] = pdu[6];
+
+ /* Pad */
+ nonce[7] = 0;
+ nonce[8] = 0;
+
+ /* IV Index */
+ sys_put_be32(iv_index, &nonce[9]);
+}
+
+int bt_mesh_net_obfuscate(u8_t *pdu, u32_t iv_index,
+ const u8_t privacy_key[16])
+{
+ u8_t priv_rand[16] = { 0x00, 0x00, 0x00, 0x00, 0x00, };
+ u8_t tmp[16];
+ int err, i;
+
+ BT_DBG("IVIndex %u, PrivacyKey %s", iv_index, bt_hex(privacy_key, 16));
+
+ sys_put_be32(iv_index, &priv_rand[5]);
+ memcpy(&priv_rand[9], &pdu[7], 7);
+
+ BT_DBG("PrivacyRandom %s", bt_hex(priv_rand, 16));
+
+ err = bt_encrypt_be(privacy_key, priv_rand, tmp);
+ if (err) {
+ return err;
+ }
+
+ for (i = 0; i < 6; i++) {
+ pdu[1 + i] ^= tmp[i];
+ }
+
+ return 0;
+}
+
+int bt_mesh_net_encrypt(const u8_t key[16], struct net_buf_simple *buf,
+ u32_t iv_index, bool proxy)
+{
+ u8_t mic_len = NET_MIC_LEN(buf->data);
+ u8_t nonce[13];
+ int err;
+
+ BT_DBG("IVIndex %u EncKey %s mic_len %u", iv_index, bt_hex(key, 16),
+ mic_len);
+ BT_DBG("PDU (len %u) %s", buf->len, bt_hex(buf->data, buf->len));
+
+#if defined(CONFIG_BT_MESH_PROXY)
+ if (proxy) {
+ create_proxy_nonce(nonce, buf->data, iv_index);
+ } else {
+ create_net_nonce(nonce, buf->data, iv_index);
+ }
+#else
+ create_net_nonce(nonce, buf->data, iv_index);
+#endif
+
+ BT_DBG("Nonce %s", bt_hex(nonce, 13));
+
+ err = bt_mesh_ccm_encrypt(key, nonce, &buf->data[7], buf->len - 7,
+ NULL, 0, &buf->data[7], mic_len);
+ if (!err) {
+ net_buf_simple_add(buf, mic_len);
+ }
+
+ return err;
+}
+
+int bt_mesh_net_decrypt(const u8_t key[16], struct net_buf_simple *buf,
+ u32_t iv_index, bool proxy)
+{
+ u8_t mic_len = NET_MIC_LEN(buf->data);
+ u8_t nonce[13];
+
+ BT_DBG("PDU (%u bytes) %s", buf->len, bt_hex(buf->data, buf->len));
+ BT_DBG("iv_index %u, key %s mic_len %u", iv_index, bt_hex(key, 16),
+ mic_len);
+
+#if defined(CONFIG_BT_MESH_PROXY)
+ if (proxy) {
+ create_proxy_nonce(nonce, buf->data, iv_index);
+ } else {
+ create_net_nonce(nonce, buf->data, iv_index);
+ }
+#else
+ create_net_nonce(nonce, buf->data, iv_index);
+#endif
+
+ BT_DBG("Nonce %s", bt_hex(nonce, 13));
+
+ buf->len -= mic_len;
+
+ return bt_mesh_ccm_decrypt(key, nonce, &buf->data[7], buf->len - 7,
+ NULL, 0, &buf->data[7], mic_len);
+}
+
+static void create_app_nonce(u8_t nonce[13], bool dev_key, u8_t aszmic,
+ u16_t src, u16_t dst, u32_t seq_num,
+ u32_t iv_index)
+{
+ if (dev_key) {
+ nonce[0] = 0x02;
+ } else {
+ nonce[0] = 0x01;
+ }
+
+ sys_put_be32((seq_num | ((u32_t)aszmic << 31)), &nonce[1]);
+
+ sys_put_be16(src, &nonce[5]);
+ sys_put_be16(dst, &nonce[7]);
+
+ sys_put_be32(iv_index, &nonce[9]);
+}
+
+int bt_mesh_app_encrypt(const u8_t key[16], bool dev_key, u8_t aszmic,
+ struct net_buf_simple *buf, const u8_t *ad,
+ u16_t src, u16_t dst, u32_t seq_num, u32_t iv_index)
+{
+ u8_t nonce[13];
+ int err;
+
+ BT_DBG("AppKey %s", bt_hex(key, 16));
+ BT_DBG("dev_key %u src 0x%04x dst 0x%04x", dev_key, src, dst);
+ BT_DBG("seq_num 0x%08x iv_index 0x%08x", seq_num, iv_index);
+ BT_DBG("Clear: %s", bt_hex(buf->data, buf->len));
+
+ create_app_nonce(nonce, dev_key, aszmic, src, dst, seq_num, iv_index);
+
+ BT_DBG("Nonce %s", bt_hex(nonce, 13));
+
+ err = bt_mesh_ccm_encrypt(key, nonce, buf->data, buf->len, ad,
+ ad ? 16 : 0, buf->data, APP_MIC_LEN(aszmic));
+ if (!err) {
+ net_buf_simple_add(buf, APP_MIC_LEN(aszmic));
+ BT_DBG("Encr: %s", bt_hex(buf->data, buf->len));
+ }
+
+ return err;
+}
+
+int bt_mesh_app_decrypt(const u8_t key[16], bool dev_key, u8_t aszmic,
+ struct net_buf_simple *buf, struct net_buf_simple *out,
+ const u8_t *ad, u16_t src, u16_t dst, u32_t seq_num,
+ u32_t iv_index)
+{
+ u8_t nonce[13];
+ int err;
+
+ BT_DBG("EncData (len %u) %s", buf->len, bt_hex(buf->data, buf->len));
+
+ create_app_nonce(nonce, dev_key, aszmic, src, dst, seq_num, iv_index);
+
+ BT_DBG("AppKey %s", bt_hex(key, 16));
+ BT_DBG("Nonce %s", bt_hex(nonce, 13));
+
+ err = bt_mesh_ccm_decrypt(key, nonce, buf->data, buf->len, ad,
+ ad ? 16 : 0, out->data, APP_MIC_LEN(aszmic));
+ if (!err) {
+ net_buf_simple_add(out, buf->len);
+ }
+
+ return err;
+}
+
+/* reversed, 8-bit, poly=0x07 */
+static const u8_t crc_table[256] = {
+ 0x00, 0x91, 0xe3, 0x72, 0x07, 0x96, 0xe4, 0x75,
+ 0x0e, 0x9f, 0xed, 0x7c, 0x09, 0x98, 0xea, 0x7b,
+ 0x1c, 0x8d, 0xff, 0x6e, 0x1b, 0x8a, 0xf8, 0x69,
+ 0x12, 0x83, 0xf1, 0x60, 0x15, 0x84, 0xf6, 0x67,
+
+ 0x38, 0xa9, 0xdb, 0x4a, 0x3f, 0xae, 0xdc, 0x4d,
+ 0x36, 0xa7, 0xd5, 0x44, 0x31, 0xa0, 0xd2, 0x43,
+ 0x24, 0xb5, 0xc7, 0x56, 0x23, 0xb2, 0xc0, 0x51,
+ 0x2a, 0xbb, 0xc9, 0x58, 0x2d, 0xbc, 0xce, 0x5f,
+
+ 0x70, 0xe1, 0x93, 0x02, 0x77, 0xe6, 0x94, 0x05,
+ 0x7e, 0xef, 0x9d, 0x0c, 0x79, 0xe8, 0x9a, 0x0b,
+ 0x6c, 0xfd, 0x8f, 0x1e, 0x6b, 0xfa, 0x88, 0x19,
+ 0x62, 0xf3, 0x81, 0x10, 0x65, 0xf4, 0x86, 0x17,
+
+ 0x48, 0xd9, 0xab, 0x3a, 0x4f, 0xde, 0xac, 0x3d,
+ 0x46, 0xd7, 0xa5, 0x34, 0x41, 0xd0, 0xa2, 0x33,
+ 0x54, 0xc5, 0xb7, 0x26, 0x53, 0xc2, 0xb0, 0x21,
+ 0x5a, 0xcb, 0xb9, 0x28, 0x5d, 0xcc, 0xbe, 0x2f,
+
+ 0xe0, 0x71, 0x03, 0x92, 0xe7, 0x76, 0x04, 0x95,
+ 0xee, 0x7f, 0x0d, 0x9c, 0xe9, 0x78, 0x0a, 0x9b,
+ 0xfc, 0x6d, 0x1f, 0x8e, 0xfb, 0x6a, 0x18, 0x89,
+ 0xf2, 0x63, 0x11, 0x80, 0xf5, 0x64, 0x16, 0x87,
+
+ 0xd8, 0x49, 0x3b, 0xaa, 0xdf, 0x4e, 0x3c, 0xad,
+ 0xd6, 0x47, 0x35, 0xa4, 0xd1, 0x40, 0x32, 0xa3,
+ 0xc4, 0x55, 0x27, 0xb6, 0xc3, 0x52, 0x20, 0xb1,
+ 0xca, 0x5b, 0x29, 0xb8, 0xcd, 0x5c, 0x2e, 0xbf,
+
+ 0x90, 0x01, 0x73, 0xe2, 0x97, 0x06, 0x74, 0xe5,
+ 0x9e, 0x0f, 0x7d, 0xec, 0x99, 0x08, 0x7a, 0xeb,
+ 0x8c, 0x1d, 0x6f, 0xfe, 0x8b, 0x1a, 0x68, 0xf9,
+ 0x82, 0x13, 0x61, 0xf0, 0x85, 0x14, 0x66, 0xf7,
+
+ 0xa8, 0x39, 0x4b, 0xda, 0xaf, 0x3e, 0x4c, 0xdd,
+ 0xa6, 0x37, 0x45, 0xd4, 0xa1, 0x30, 0x42, 0xd3,
+ 0xb4, 0x25, 0x57, 0xc6, 0xb3, 0x22, 0x50, 0xc1,
+ 0xba, 0x2b, 0x59, 0xc8, 0xbd, 0x2c, 0x5e, 0xcf
+};
+
+u8_t bt_mesh_fcs_calc(const u8_t *data, u8_t data_len)
+{
+ u8_t fcs = 0xff;
+
+ while (data_len--) {
+ fcs = crc_table[fcs ^ *data++];
+ }
+
+ BT_DBG("fcs 0x%02x", 0xff - fcs);
+
+ return 0xff - fcs;
+}
+
+bool bt_mesh_fcs_check(struct net_buf_simple *buf, u8_t received_fcs)
+{
+ const u8_t *data = buf->data;
+ u16_t data_len = buf->len;
+ u8_t fcs = 0xff;
+
+ while (data_len--) {
+ fcs = crc_table[fcs ^ *data++];
+ }
+
+ return crc_table[fcs ^ received_fcs] == 0xcf;
+}
+
+int bt_mesh_virtual_addr(const u8_t virtual_label[16], u16_t *addr)
+{
+ u8_t salt[16];
+ u8_t tmp[16];
+ int err;
+
+ err = bt_mesh_s1("vtad", salt);
+ if (err) {
+ return err;
+ }
+
+ err = bt_mesh_aes_cmac_one(salt, virtual_label, 16, tmp);
+ if (err) {
+ return err;
+ }
+
+ *addr = (sys_get_be16(&tmp[14]) & 0x3fff) | 0x8000;
+
+ return 0;
+}
+
+int bt_mesh_prov_conf_salt(const u8_t conf_inputs[145], u8_t salt[16])
+{
+ const u8_t conf_salt_key[16] = { 0 };
+
+ return bt_mesh_aes_cmac_one(conf_salt_key, conf_inputs, 145, salt);
+}
+
+int bt_mesh_prov_conf_key(const u8_t dhkey[32], const u8_t conf_salt[16],
+ u8_t conf_key[16])
+{
+ return bt_mesh_k1(dhkey, 32, conf_salt, "prck", conf_key);
+}
+
+int bt_mesh_prov_conf(const u8_t conf_key[16], const u8_t rand[16],
+ const u8_t auth[16], u8_t conf[16])
+{
+ struct bt_mesh_sg sg[] = { { rand, 16 }, { auth, 16 } };
+
+ BT_DBG("ConfirmationKey %s", bt_hex(conf_key, 16));
+ BT_DBG("RandomDevice %s", bt_hex(rand, 16));
+ BT_DBG("AuthValue %s", bt_hex(auth, 16));
+
+ return bt_mesh_aes_cmac(conf_key, sg, ARRAY_SIZE(sg), conf);
+}
+
+int bt_mesh_prov_decrypt(const u8_t key[16], u8_t nonce[13],
+ const u8_t data[25 + 8], u8_t out[25])
+{
+ return bt_mesh_ccm_decrypt(key, nonce, data, 25, NULL, 0, out, 8);
+}
+
+#if CONFIG_BT_MESH_PROVISIONER
+int bt_mesh_prov_encrypt(const u8_t key[16], u8_t nonce[13],
+ const u8_t data[25], u8_t out[33])
+{
+ return bt_mesh_ccm_encrypt(key, nonce, data, 25, NULL, 0, out, 8);
+}
+#endif
+
+int bt_mesh_beacon_auth(const u8_t beacon_key[16], u8_t flags,
+ const u8_t net_id[8], u32_t iv_index,
+ u8_t auth[8])
+{
+ u8_t msg[13], tmp[16];
+ int err;
+
+ BT_DBG("BeaconKey %s", bt_hex(beacon_key, 16));
+ BT_DBG("NetId %s", bt_hex(net_id, 8));
+ BT_DBG("IV Index 0x%08x", iv_index);
+
+ msg[0] = flags;
+ memcpy(&msg[1], net_id, 8);
+ sys_put_be32(iv_index, &msg[9]);
+
+ BT_DBG("BeaconMsg %s", bt_hex(msg, sizeof(msg)));
+
+ err = bt_mesh_aes_cmac_one(beacon_key, msg, sizeof(msg), tmp);
+ if (!err) {
+ memcpy(auth, tmp, 8);
+ }
+
+ return err;
+}
+
+#endif /* #if CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/crypto.h b/components/bt/ble_mesh/mesh_core/crypto.h
new file mode 100644
index 0000000000..314872e327
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/crypto.h
@@ -0,0 +1,163 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "mesh_types.h"
+#include
+
+struct bt_mesh_sg {
+ const void *data;
+ size_t len;
+};
+
+int bt_mesh_aes_cmac(const u8_t key[16], struct bt_mesh_sg *sg,
+ size_t sg_len, u8_t mac[16]);
+
+static inline int bt_mesh_aes_cmac_one(const u8_t key[16], const void *m,
+ size_t len, u8_t mac[16])
+{
+ struct bt_mesh_sg sg = { m, len };
+
+ return bt_mesh_aes_cmac(key, &sg, 1, mac);
+}
+
+static inline bool bt_mesh_s1(const char *m, u8_t salt[16])
+{
+ const u8_t zero[16] = { 0 };
+
+ return bt_mesh_aes_cmac_one(zero, m, strlen(m), salt);
+}
+
+int bt_mesh_k1(const u8_t *ikm, size_t ikm_len, const u8_t salt[16],
+ const char *info, u8_t okm[16]);
+
+#define bt_mesh_k1_str(ikm, ikm_len, salt_str, info, okm) \
+({ \
+ const u8_t salt[16] = salt_str; \
+ bt_mesh_k1(ikm, ikm_len, salt, info, okm); \
+})
+
+int bt_mesh_k2(const u8_t n[16], const u8_t *p, size_t p_len,
+ u8_t net_id[1], u8_t enc_key[16], u8_t priv_key[16]);
+
+int bt_mesh_k3(const u8_t n[16], u8_t out[8]);
+
+int bt_mesh_k4(const u8_t n[16], u8_t out[1]);
+
+int bt_mesh_id128(const u8_t n[16], const char *s, u8_t out[16]);
+
+static inline int bt_mesh_id_resolving_key(const u8_t net_key[16],
+ u8_t resolving_key[16])
+{
+ return bt_mesh_k1_str(net_key, 16, "smbt", "smbi", resolving_key);
+}
+
+static inline int bt_mesh_identity_key(const u8_t net_key[16],
+ u8_t identity_key[16])
+{
+ return bt_mesh_id128(net_key, "nkik", identity_key);
+}
+
+static inline int bt_mesh_beacon_key(const u8_t net_key[16],
+ u8_t beacon_key[16])
+{
+ return bt_mesh_id128(net_key, "nkbk", beacon_key);
+}
+
+int bt_mesh_beacon_auth(const u8_t beacon_key[16], u8_t flags,
+ const u8_t net_id[16], u32_t iv_index,
+ u8_t auth[8]);
+
+static inline int bt_mesh_app_id(const u8_t app_key[16], u8_t app_id[1])
+{
+ return bt_mesh_k4(app_key, app_id);
+}
+
+static inline int bt_mesh_session_key(const u8_t dhkey[32],
+ const u8_t prov_salt[16],
+ u8_t session_key[16])
+{
+ return bt_mesh_k1(dhkey, 32, prov_salt, "prsk", session_key);
+}
+
+static inline int bt_mesh_prov_nonce(const u8_t dhkey[32],
+ const u8_t prov_salt[16],
+ u8_t nonce[13])
+{
+ u8_t tmp[16];
+ int err;
+
+ err = bt_mesh_k1(dhkey, 32, prov_salt, "prsn", tmp);
+ if (!err) {
+ memcpy(nonce, tmp + 3, 13);
+ }
+
+ return err;
+}
+
+static inline int bt_mesh_dev_key(const u8_t dhkey[32],
+ const u8_t prov_salt[16],
+ u8_t dev_key[16])
+{
+ return bt_mesh_k1(dhkey, 32, prov_salt, "prdk", dev_key);
+}
+
+static inline int bt_mesh_prov_salt(const u8_t conf_salt[16],
+ const u8_t prov_rand[16],
+ const u8_t dev_rand[16],
+ u8_t prov_salt[16])
+{
+ const u8_t prov_salt_key[16] = { 0 };
+ struct bt_mesh_sg sg[] = {
+ { conf_salt, 16 },
+ { prov_rand, 16 },
+ { dev_rand, 16 },
+ };
+
+ return bt_mesh_aes_cmac(prov_salt_key, sg, ARRAY_SIZE(sg), prov_salt);
+}
+
+int bt_mesh_net_obfuscate(u8_t *pdu, u32_t iv_index,
+ const u8_t privacy_key[16]);
+
+int bt_mesh_net_encrypt(const u8_t key[16], struct net_buf_simple *buf,
+ u32_t iv_index, bool proxy);
+
+int bt_mesh_net_decrypt(const u8_t key[16], struct net_buf_simple *buf,
+ u32_t iv_index, bool proxy);
+
+int bt_mesh_app_encrypt(const u8_t key[16], bool dev_key, u8_t aszmic,
+ struct net_buf_simple *buf, const u8_t *ad,
+ u16_t src, u16_t dst, u32_t seq_num, u32_t iv_index);
+
+int bt_mesh_app_decrypt(const u8_t key[16], bool dev_key, u8_t aszmic,
+ struct net_buf_simple *buf, struct net_buf_simple *out,
+ const u8_t *ad, u16_t src, u16_t dst, u32_t seq_num,
+ u32_t iv_index);
+
+u8_t bt_mesh_fcs_calc(const u8_t *data, u8_t data_len);
+
+bool bt_mesh_fcs_check(struct net_buf_simple *buf, u8_t received_fcs);
+
+int bt_mesh_virtual_addr(const u8_t virtual_label[16], u16_t *addr);
+
+int bt_mesh_prov_conf_salt(const u8_t conf_inputs[145], u8_t salt[16]);
+
+int bt_mesh_prov_conf_key(const u8_t dhkey[32], const u8_t conf_salt[16],
+ u8_t conf_key[16]);
+
+int bt_mesh_prov_conf(const u8_t conf_key[16], const u8_t rand[16],
+ const u8_t auth[16], u8_t conf[16]);
+
+int bt_mesh_prov_decrypt(const u8_t key[16], u8_t nonce[13],
+ const u8_t data[25 + 8], u8_t out[25]);
+
+#if CONFIG_BT_MESH_PROVISIONER
+int bt_mesh_prov_encrypt(const u8_t key[16], u8_t nonce[13],
+ const u8_t data[25], u8_t out[33]);
+#endif
diff --git a/components/bt/ble_mesh/mesh_core/dlist.c b/components/bt/ble_mesh/mesh_core/dlist.c
new file mode 100644
index 0000000000..e690a0c291
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/dlist.c
@@ -0,0 +1,319 @@
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "dlist.h"
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+
+static sys_dlist_t test_list;
+
+struct container_node {
+ sys_dnode_t node;
+ int unused;
+};
+
+static struct container_node test_node_1;
+static struct container_node test_node_2;
+static struct container_node test_node_3;
+static struct container_node test_node_4;
+
+static inline bool verify_emptyness(sys_dlist_t *list)
+{
+ sys_dnode_t *node;
+ sys_dnode_t *s_node;
+ struct container_node *cnode;
+ struct container_node *s_cnode;
+ int count;
+
+ if (!sys_dlist_is_empty(list)) {
+ return false;
+ }
+
+ if (sys_dlist_peek_head(list)) {
+ return false;
+ }
+
+ if (sys_dlist_peek_tail(list)) {
+ return false;
+ }
+
+ count = 0;
+ SYS_DLIST_FOR_EACH_NODE(list, node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ SYS_DLIST_FOR_EACH_NODE_SAFE(list, node, s_node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ count = 0;
+ SYS_DLIST_FOR_EACH_CONTAINER(list, cnode, node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ count = 0;
+ SYS_DLIST_FOR_EACH_CONTAINER_SAFE(list, cnode, s_cnode, node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ return true;
+}
+
+static inline bool verify_content_amount(sys_dlist_t *list, int amount)
+{
+ sys_dnode_t *node;
+ sys_dnode_t *s_node;
+ struct container_node *cnode;
+ struct container_node *s_cnode;
+ int count;
+
+ if (sys_dlist_is_empty(list)) {
+ return false;
+ }
+
+ if (!sys_dlist_peek_head(list)) {
+ return false;
+ }
+
+ if (!sys_dlist_peek_tail(list)) {
+ return false;
+ }
+
+ count = 0;
+ SYS_DLIST_FOR_EACH_NODE(list, node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ count = 0;
+ SYS_DLIST_FOR_EACH_NODE_SAFE(list, node, s_node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ count = 0;
+ SYS_DLIST_FOR_EACH_CONTAINER(list, cnode, node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ count = 0;
+ SYS_DLIST_FOR_EACH_CONTAINER_SAFE(list, cnode, s_cnode, node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ return true;
+}
+
+static inline bool verify_tail_head(sys_dlist_t *list,
+ sys_dnode_t *head,
+ sys_dnode_t *tail,
+ bool same)
+{
+ if (sys_dlist_peek_head(list) != head) {
+ return false;
+ }
+
+ if (sys_dlist_peek_tail(list) != tail) {
+ return false;
+ }
+
+ if (same) {
+ if (sys_dlist_peek_head(list) != sys_dlist_peek_tail(list)) {
+ return false;
+ }
+ } else {
+ if (sys_dlist_peek_head(list) == sys_dlist_peek_tail(list)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+void dlist_test(void)
+{
+ sys_dlist_init(&test_list);
+
+ //zassert_true((verify_emptyness(&test_list)), "test_list should be empty");
+
+ /* Appending node 1 */
+ sys_dlist_append(&test_list, &test_node_1.node);
+ //zassert_true((verify_content_amount(&test_list, 1)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_1.node,
+ // &test_node_1.node, true)),
+ // "test_list head/tail are wrong");
+
+ /* Finding and removing node 1 */
+ sys_dlist_remove(&test_node_1.node);
+ //zassert_true((verify_emptyness(&test_list)),
+ // "test_list should be empty");
+
+ /* Prepending node 1 */
+ sys_dlist_prepend(&test_list, &test_node_1.node);
+ //zassert_true((verify_content_amount(&test_list, 1)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_1.node,
+ // &test_node_1.node, true)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 1 */
+ sys_dlist_remove(&test_node_1.node);
+ //zassert_true((verify_emptyness(&test_list)),
+ // "test_list should be empty");
+
+ /* Appending node 1 */
+ sys_dlist_append(&test_list, &test_node_1.node);
+ /* Prepending node 2 */
+ sys_dlist_prepend(&test_list, &test_node_2.node);
+
+ //zassert_true((verify_content_amount(&test_list, 2)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_1.node, false)),
+ // "test_list head/tail are wrong");
+
+ /* Appending node 3 */
+ sys_dlist_append(&test_list, &test_node_3.node);
+
+ //zassert_true((verify_content_amount(&test_list, 3)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_3.node, false)),
+ // "test_list head/tail are wrong");
+
+ //zassert_true((sys_dlist_peek_next(&test_list, &test_node_2.node) ==
+ // &test_node_1.node),
+ // "test_list node links are wrong");
+
+ /* Inserting node 4 after node 2 */
+ sys_dlist_insert_after(&test_list, &test_node_2.node,
+ &test_node_4.node);
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_3.node, false)),
+ // "test_list head/tail are wrong");
+
+ //zassert_true((sys_dlist_peek_next(&test_list, &test_node_2.node) ==
+ // &test_node_4.node),
+ // "test_list node links are wrong");
+
+ /* Finding and removing node 1 */
+ sys_dlist_remove(&test_node_1.node);
+ //zassert_true((verify_content_amount(&test_list, 3)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_3.node, false)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 3 */
+ sys_dlist_remove(&test_node_3.node);
+ //zassert_true((verify_content_amount(&test_list, 2)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_4.node, false)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 4 */
+ sys_dlist_remove(&test_node_4.node);
+ //zassert_true((verify_content_amount(&test_list, 1)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_2.node, true)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 2 */
+ sys_dlist_remove(&test_node_2.node);
+ //zassert_true((verify_emptyness(&test_list)),
+ // "test_list should be empty");
+
+ /* test iterator from a node */
+ struct data_node {
+ sys_dnode_t node;
+ int data;
+ } data_node[6] = {
+ { .data = 0 },
+ { .data = 1 },
+ { .data = 2 },
+ { .data = 3 },
+ { .data = 4 },
+ { .data = 5 },
+ };
+ sys_dnode_t *node = NULL;
+ int ii;
+
+ sys_dlist_init(&test_list);
+
+ for (ii = 0; ii < 6; ii++) {
+ sys_dlist_append(&test_list, &data_node[ii].node);
+ }
+
+ ii = 0;
+ SYS_DLIST_ITERATE_FROM_NODE(&test_list, node) {
+ ii++;
+ if (((struct data_node *)node)->data == 2) {
+ break;
+ }
+ }
+ //zassert_equal(ii, 3, "");
+
+ ii = 0;
+ SYS_DLIST_ITERATE_FROM_NODE(&test_list, node) {
+ ii++;
+ if (((struct data_node *)node)->data == 3) {
+ break;
+ }
+ }
+ //zassert_equal(ii, 1, "");
+
+ ii = 0;
+ SYS_DLIST_ITERATE_FROM_NODE(&test_list, node) {
+ ii++;
+ }
+ //zassert_equal(ii, 2, "");
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/mesh_core/dlist.h b/components/bt/ble_mesh/mesh_core/dlist.h
new file mode 100644
index 0000000000..0ad75a0d52
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/dlist.h
@@ -0,0 +1,496 @@
+/*
+ * Copyright (c) 2013-2015 Wind River Systems, Inc.
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+/**
+ * @file
+ * @brief Doubly-linked list implementation
+ *
+ * Doubly-linked list implementation using inline macros/functions.
+ * This API is not thread safe, and thus if a list is used across threads,
+ * calls to functions must be protected with synchronization primitives.
+ *
+ * The lists are expected to be initialized such that both the head and tail
+ * pointers point to the list itself. Initializing the lists in such a fashion
+ * simplifies the adding and removing of nodes to/from the list.
+ */
+
+#ifndef _misc_dlist__h_
+#define _misc_dlist__h_
+
+#include
+#include "mesh_util.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+struct _dnode {
+ union {
+ struct _dnode *head; /* ptr to head of list (sys_dlist_t) */
+ struct _dnode *next; /* ptr to next node (sys_dnode_t) */
+ };
+ union {
+ struct _dnode *tail; /* ptr to tail of list (sys_dlist_t) */
+ struct _dnode *prev; /* ptr to previous node (sys_dnode_t) */
+ };
+};
+
+typedef struct _dnode sys_dlist_t;
+typedef struct _dnode sys_dnode_t;
+
+/**
+ * @brief Provide the primitive to iterate on a list
+ * Note: the loop is unsafe and thus __dn should not be removed
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_NODE(l, n) {
+ *
+ * }
+ *
+ * This and other SYS_DLIST_*() macros are not thread safe.
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __dn A sys_dnode_t pointer to peek each node of the list
+ */
+#define SYS_DLIST_FOR_EACH_NODE(__dl, __dn) \
+ for (__dn = sys_dlist_peek_head(__dl); __dn; \
+ __dn = sys_dlist_peek_next(__dl, __dn))
+
+/**
+ * @brief Provide the primitive to iterate on a list, from a node in the list
+ * Note: the loop is unsafe and thus __dn should not be removed
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_ITERATE_FROM_NODE(l, n) {
+ *
+ * }
+ *
+ * Like SYS_DLIST_FOR_EACH_NODE(), but __dn already contains a node in the list
+ * where to start searching for the next entry from. If NULL, it starts from
+ * the head.
+ *
+ * This and other SYS_DLIST_*() macros are not thread safe.
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __dn A sys_dnode_t pointer to peek each node of the list;
+ * it contains the starting node, or NULL to start from the head
+ */
+#define SYS_DLIST_ITERATE_FROM_NODE(__dl, __dn) \
+ for (__dn = __dn ? sys_dlist_peek_next_no_check(__dl, __dn) \
+ : sys_dlist_peek_head(__dl); \
+ __dn; \
+ __dn = sys_dlist_peek_next(__dl, __dn))
+
+/**
+ * @brief Provide the primitive to safely iterate on a list
+ * Note: __dn can be removed, it will not break the loop.
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_NODE_SAFE(l, n, s) {
+ *
+ * }
+ *
+ * This and other SYS_DLIST_*() macros are not thread safe.
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __dn A sys_dnode_t pointer to peek each node of the list
+ * @param __dns A sys_dnode_t pointer for the loop to run safely
+ */
+#define SYS_DLIST_FOR_EACH_NODE_SAFE(__dl, __dn, __dns) \
+ for (__dn = sys_dlist_peek_head(__dl), \
+ __dns = sys_dlist_peek_next(__dl, __dn); \
+ __dn; __dn = __dns, \
+ __dns = sys_dlist_peek_next(__dl, __dn))
+
+/*
+ * @brief Provide the primitive to resolve the container of a list node
+ * Note: it is safe to use with NULL pointer nodes
+ *
+ * @param __dn A pointer on a sys_dnode_t to get its container
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_CONTAINER(__dn, __cn, __n) \
+ (__dn ? CONTAINER_OF(__dn, __typeof__(*__cn), __n) : NULL)
+/*
+ * @brief Provide the primitive to peek container of the list head
+ *
+ * @param __dl A pointer on a sys_dlist_t to peek
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n) \
+ SYS_DLIST_CONTAINER(sys_dlist_peek_head(__dl), __cn, __n)
+
+/*
+ * @brief Provide the primitive to peek the next container
+ *
+ * @param __dl A pointer on a sys_dlist_t to peek
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n) \
+ ((__cn) ? SYS_DLIST_CONTAINER(sys_dlist_peek_next(__dl, &(__cn->__n)), \
+ __cn, __n) : NULL)
+
+/**
+ * @brief Provide the primitive to iterate on a list under a container
+ * Note: the loop is unsafe and thus __cn should not be detached
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_CONTAINER(l, c, n) {
+ *
+ * }
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __cn A pointer to peek each entry of the list
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_FOR_EACH_CONTAINER(__dl, __cn, __n) \
+ for (__cn = SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n); __cn; \
+ __cn = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n))
+
+/**
+ * @brief Provide the primitive to safely iterate on a list under a container
+ * Note: __cn can be detached, it will not break the loop.
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_CONTAINER_SAFE(l, c, cn, n) {
+ *
+ * }
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __cn A pointer to peek each entry of the list
+ * @param __cns A pointer for the loop to run safely
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_FOR_EACH_CONTAINER_SAFE(__dl, __cn, __cns, __n) \
+ for (__cn = SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n), \
+ __cns = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n); __cn; \
+ __cn = __cns, \
+ __cns = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n))
+
+/**
+ * @brief initialize list
+ *
+ * @param list the doubly-linked list
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_init(sys_dlist_t *list)
+{
+ list->head = (sys_dnode_t *)list;
+ list->tail = (sys_dnode_t *)list;
+}
+
+#define SYS_DLIST_STATIC_INIT(ptr_to_list) {{(ptr_to_list)}, {(ptr_to_list)}}
+
+/**
+ * @brief check if a node is the list's head
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node to check
+ *
+ * @return 1 if node is the head, 0 otherwise
+ */
+
+static inline int sys_dlist_is_head(sys_dlist_t *list, sys_dnode_t *node)
+{
+ return list->head == node;
+}
+
+/**
+ * @brief check if a node is the list's tail
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node to check
+ *
+ * @return 1 if node is the tail, 0 otherwise
+ */
+
+static inline int sys_dlist_is_tail(sys_dlist_t *list, sys_dnode_t *node)
+{
+ return list->tail == node;
+}
+
+/**
+ * @brief check if the list is empty
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return 1 if empty, 0 otherwise
+ */
+
+static inline int sys_dlist_is_empty(sys_dlist_t *list)
+{
+ return list->head == list;
+}
+
+/**
+ * @brief check if more than one node present
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return 1 if multiple nodes, 0 otherwise
+ */
+
+static inline int sys_dlist_has_multiple_nodes(sys_dlist_t *list)
+{
+ return list->head != list->tail;
+}
+
+/**
+ * @brief get a reference to the head item in the list
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return a pointer to the head element, NULL if list is empty
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_head(sys_dlist_t *list)
+{
+ return sys_dlist_is_empty(list) ? NULL : list->head;
+}
+
+/**
+ * @brief get a reference to the head item in the list
+ *
+ * The list must be known to be non-empty.
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return a pointer to the head element
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_head_not_empty(sys_dlist_t *list)
+{
+ return list->head;
+}
+
+/**
+ * @brief get a reference to the next item in the list, node is not NULL
+ *
+ * Faster than sys_dlist_peek_next() if node is known not to be NULL.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node from which to get the next element in the list
+ *
+ * @return a pointer to the next element from a node, NULL if node is the tail
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_next_no_check(sys_dlist_t *list,
+ sys_dnode_t *node)
+{
+ return (node == list->tail) ? NULL : node->next;
+}
+
+/**
+ * @brief get a reference to the next item in the list
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node from which to get the next element in the list
+ *
+ * @return a pointer to the next element from a node, NULL if node is the tail
+ * or NULL (when node comes from reading the head of an empty list).
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_next(sys_dlist_t *list,
+ sys_dnode_t *node)
+{
+ return node ? sys_dlist_peek_next_no_check(list, node) : NULL;
+}
+
+/**
+ * @brief get a reference to the tail item in the list
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return a pointer to the tail element, NULL if list is empty
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_tail(sys_dlist_t *list)
+{
+ return sys_dlist_is_empty(list) ? NULL : list->tail;
+}
+
+/**
+ * @brief add node to tail of list
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the element to append
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_append(sys_dlist_t *list, sys_dnode_t *node)
+{
+ node->next = list;
+ node->prev = list->tail;
+
+ list->tail->next = node;
+ list->tail = node;
+}
+
+/**
+ * @brief add node to head of list
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the element to append
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_prepend(sys_dlist_t *list, sys_dnode_t *node)
+{
+ node->next = list->head;
+ node->prev = list;
+
+ list->head->prev = node;
+ list->head = node;
+}
+
+/**
+ * @brief insert node after a node
+ *
+ * Insert a node after a specified node in a list.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param insert_point the insert point in the list: if NULL, insert at head
+ * @param node the element to append
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_insert_after(sys_dlist_t *list,
+ sys_dnode_t *insert_point, sys_dnode_t *node)
+{
+ if (!insert_point) {
+ sys_dlist_prepend(list, node);
+ } else {
+ node->next = insert_point->next;
+ node->prev = insert_point;
+ insert_point->next->prev = node;
+ insert_point->next = node;
+ }
+}
+
+/**
+ * @brief insert node before a node
+ *
+ * Insert a node before a specified node in a list.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param insert_point the insert point in the list: if NULL, insert at tail
+ * @param node the element to insert
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_insert_before(sys_dlist_t *list,
+ sys_dnode_t *insert_point, sys_dnode_t *node)
+{
+ if (!insert_point) {
+ sys_dlist_append(list, node);
+ } else {
+ node->prev = insert_point->prev;
+ node->next = insert_point;
+ insert_point->prev->next = node;
+ insert_point->prev = node;
+ }
+}
+
+/**
+ * @brief insert node at position
+ *
+ * Insert a node in a location depending on a external condition. The cond()
+ * function checks if the node is to be inserted _before_ the current node
+ * against which it is checked.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the element to insert
+ * @param cond a function that determines if the current node is the correct
+ * insert point
+ * @param data parameter to cond()
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_insert_at(sys_dlist_t *list, sys_dnode_t *node,
+ int (*cond)(sys_dnode_t *, void *), void *data)
+{
+ if (sys_dlist_is_empty(list)) {
+ sys_dlist_append(list, node);
+ } else {
+ sys_dnode_t *pos = sys_dlist_peek_head(list);
+
+ while (pos && !cond(pos, data)) {
+ pos = sys_dlist_peek_next(list, pos);
+ }
+ sys_dlist_insert_before(list, pos, node);
+ }
+}
+
+/**
+ * @brief remove a specific node from a list
+ *
+ * The list is implicit from the node. The node must be part of a list.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param node the node to remove
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_remove(sys_dnode_t *node)
+{
+ node->prev->next = node->next;
+ node->next->prev = node->prev;
+}
+
+/**
+ * @brief get the first node in a list
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return the first node in the list, NULL if list is empty
+ */
+
+static inline sys_dnode_t *sys_dlist_get(sys_dlist_t *list)
+{
+ sys_dnode_t *node;
+
+ if (sys_dlist_is_empty(list)) {
+ return NULL;
+ }
+
+ node = list->head;
+ sys_dlist_remove(node);
+ return node;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* _misc_dlist__h_ */
+
diff --git a/components/bt/ble_mesh/mesh_core/foundation.h b/components/bt/ble_mesh/mesh_core/foundation.h
new file mode 100644
index 0000000000..de90b71a4c
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/foundation.h
@@ -0,0 +1,165 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef _FOUNDATION_H_
+#define _FOUNDATION_H_
+#include "mesh_access.h"
+#include "net.h"
+
+#define OP_APP_KEY_ADD BT_MESH_MODEL_OP_1(0x00)
+#define OP_APP_KEY_UPDATE BT_MESH_MODEL_OP_1(0x01)
+#define OP_DEV_COMP_DATA_STATUS BT_MESH_MODEL_OP_1(0x02)
+#define OP_MOD_PUB_SET BT_MESH_MODEL_OP_1(0x03)
+#define OP_HEALTH_CURRENT_STATUS BT_MESH_MODEL_OP_1(0x04)
+#define OP_HEALTH_FAULT_STATUS BT_MESH_MODEL_OP_1(0x05)
+#define OP_HEARTBEAT_PUB_STATUS BT_MESH_MODEL_OP_1(0x06)
+#define OP_APP_KEY_DEL BT_MESH_MODEL_OP_2(0x80, 0x00)
+#define OP_APP_KEY_GET BT_MESH_MODEL_OP_2(0x80, 0x01)
+#define OP_APP_KEY_LIST BT_MESH_MODEL_OP_2(0x80, 0x02)
+#define OP_APP_KEY_STATUS BT_MESH_MODEL_OP_2(0x80, 0x03)
+#define OP_ATTENTION_GET BT_MESH_MODEL_OP_2(0x80, 0x04)
+#define OP_ATTENTION_SET BT_MESH_MODEL_OP_2(0x80, 0x05)
+#define OP_ATTENTION_SET_UNREL BT_MESH_MODEL_OP_2(0x80, 0x06)
+#define OP_ATTENTION_STATUS BT_MESH_MODEL_OP_2(0x80, 0x07)
+#define OP_DEV_COMP_DATA_GET BT_MESH_MODEL_OP_2(0x80, 0x08)
+#define OP_BEACON_GET BT_MESH_MODEL_OP_2(0x80, 0x09)
+#define OP_BEACON_SET BT_MESH_MODEL_OP_2(0x80, 0x0a)
+#define OP_BEACON_STATUS BT_MESH_MODEL_OP_2(0x80, 0x0b)
+#define OP_DEFAULT_TTL_GET BT_MESH_MODEL_OP_2(0x80, 0x0c)
+#define OP_DEFAULT_TTL_SET BT_MESH_MODEL_OP_2(0x80, 0x0d)
+#define OP_DEFAULT_TTL_STATUS BT_MESH_MODEL_OP_2(0x80, 0x0e)
+#define OP_FRIEND_GET BT_MESH_MODEL_OP_2(0x80, 0x0f)
+#define OP_FRIEND_SET BT_MESH_MODEL_OP_2(0x80, 0x10)
+#define OP_FRIEND_STATUS BT_MESH_MODEL_OP_2(0x80, 0x11)
+#define OP_GATT_PROXY_GET BT_MESH_MODEL_OP_2(0x80, 0x12)
+#define OP_GATT_PROXY_SET BT_MESH_MODEL_OP_2(0x80, 0x13)
+#define OP_GATT_PROXY_STATUS BT_MESH_MODEL_OP_2(0x80, 0x14)
+#define OP_KRP_GET BT_MESH_MODEL_OP_2(0x80, 0x15)
+#define OP_KRP_SET BT_MESH_MODEL_OP_2(0x80, 0x16)
+#define OP_KRP_STATUS BT_MESH_MODEL_OP_2(0x80, 0x17)
+#define OP_MOD_PUB_GET BT_MESH_MODEL_OP_2(0x80, 0x18)
+#define OP_MOD_PUB_STATUS BT_MESH_MODEL_OP_2(0x80, 0x19)
+#define OP_MOD_PUB_VA_SET BT_MESH_MODEL_OP_2(0x80, 0x1a)
+#define OP_MOD_SUB_ADD BT_MESH_MODEL_OP_2(0x80, 0x1b)
+#define OP_MOD_SUB_DEL BT_MESH_MODEL_OP_2(0x80, 0x1c)
+#define OP_MOD_SUB_DEL_ALL BT_MESH_MODEL_OP_2(0x80, 0x1d)
+#define OP_MOD_SUB_OVERWRITE BT_MESH_MODEL_OP_2(0x80, 0x1e)
+#define OP_MOD_SUB_STATUS BT_MESH_MODEL_OP_2(0x80, 0x1f)
+#define OP_MOD_SUB_VA_ADD BT_MESH_MODEL_OP_2(0x80, 0x20)
+#define OP_MOD_SUB_VA_DEL BT_MESH_MODEL_OP_2(0x80, 0x21)
+#define OP_MOD_SUB_VA_OVERWRITE BT_MESH_MODEL_OP_2(0x80, 0x22)
+#define OP_NET_TRANSMIT_GET BT_MESH_MODEL_OP_2(0x80, 0x23)
+#define OP_NET_TRANSMIT_SET BT_MESH_MODEL_OP_2(0x80, 0x24)
+#define OP_NET_TRANSMIT_STATUS BT_MESH_MODEL_OP_2(0x80, 0x25)
+#define OP_RELAY_GET BT_MESH_MODEL_OP_2(0x80, 0x26)
+#define OP_RELAY_SET BT_MESH_MODEL_OP_2(0x80, 0x27)
+#define OP_RELAY_STATUS BT_MESH_MODEL_OP_2(0x80, 0x28)
+#define OP_MOD_SUB_GET BT_MESH_MODEL_OP_2(0x80, 0x29)
+#define OP_MOD_SUB_LIST BT_MESH_MODEL_OP_2(0x80, 0x2a)
+#define OP_MOD_SUB_GET_VND BT_MESH_MODEL_OP_2(0x80, 0x2b)
+#define OP_MOD_SUB_LIST_VND BT_MESH_MODEL_OP_2(0x80, 0x2c)
+#define OP_LPN_TIMEOUT_GET BT_MESH_MODEL_OP_2(0x80, 0x2d)
+#define OP_LPN_TIMEOUT_STATUS BT_MESH_MODEL_OP_2(0x80, 0x2e)
+#define OP_HEALTH_FAULT_CLEAR BT_MESH_MODEL_OP_2(0x80, 0x2f)
+#define OP_HEALTH_FAULT_CLEAR_UNREL BT_MESH_MODEL_OP_2(0x80, 0x30)
+#define OP_HEALTH_FAULT_GET BT_MESH_MODEL_OP_2(0x80, 0x31)
+#define OP_HEALTH_FAULT_TEST BT_MESH_MODEL_OP_2(0x80, 0x32)
+#define OP_HEALTH_FAULT_TEST_UNREL BT_MESH_MODEL_OP_2(0x80, 0x33)
+#define OP_HEALTH_PERIOD_GET BT_MESH_MODEL_OP_2(0x80, 0x34)
+#define OP_HEALTH_PERIOD_SET BT_MESH_MODEL_OP_2(0x80, 0x35)
+#define OP_HEALTH_PERIOD_SET_UNREL BT_MESH_MODEL_OP_2(0x80, 0x36)
+#define OP_HEALTH_PERIOD_STATUS BT_MESH_MODEL_OP_2(0x80, 0x37)
+#define OP_HEARTBEAT_PUB_GET BT_MESH_MODEL_OP_2(0x80, 0x38)
+#define OP_HEARTBEAT_PUB_SET BT_MESH_MODEL_OP_2(0x80, 0x39)
+#define OP_HEARTBEAT_SUB_GET BT_MESH_MODEL_OP_2(0x80, 0x3a)
+#define OP_HEARTBEAT_SUB_SET BT_MESH_MODEL_OP_2(0x80, 0x3b)
+#define OP_HEARTBEAT_SUB_STATUS BT_MESH_MODEL_OP_2(0x80, 0x3c)
+#define OP_MOD_APP_BIND BT_MESH_MODEL_OP_2(0x80, 0x3d)
+#define OP_MOD_APP_STATUS BT_MESH_MODEL_OP_2(0x80, 0x3e)
+#define OP_MOD_APP_UNBIND BT_MESH_MODEL_OP_2(0x80, 0x3f)
+#define OP_NET_KEY_ADD BT_MESH_MODEL_OP_2(0x80, 0x40)
+#define OP_NET_KEY_DEL BT_MESH_MODEL_OP_2(0x80, 0x41)
+#define OP_NET_KEY_GET BT_MESH_MODEL_OP_2(0x80, 0x42)
+#define OP_NET_KEY_LIST BT_MESH_MODEL_OP_2(0x80, 0x43)
+#define OP_NET_KEY_STATUS BT_MESH_MODEL_OP_2(0x80, 0x44)
+#define OP_NET_KEY_UPDATE BT_MESH_MODEL_OP_2(0x80, 0x45)
+#define OP_NODE_IDENTITY_GET BT_MESH_MODEL_OP_2(0x80, 0x46)
+#define OP_NODE_IDENTITY_SET BT_MESH_MODEL_OP_2(0x80, 0x47)
+#define OP_NODE_IDENTITY_STATUS BT_MESH_MODEL_OP_2(0x80, 0x48)
+#define OP_NODE_RESET BT_MESH_MODEL_OP_2(0x80, 0x49)
+#define OP_NODE_RESET_STATUS BT_MESH_MODEL_OP_2(0x80, 0x4a)
+#define OP_SIG_MOD_APP_GET BT_MESH_MODEL_OP_2(0x80, 0x4b)
+#define OP_SIG_MOD_APP_LIST BT_MESH_MODEL_OP_2(0x80, 0x4c)
+#define OP_VND_MOD_APP_GET BT_MESH_MODEL_OP_2(0x80, 0x4d)
+#define OP_VND_MOD_APP_LIST BT_MESH_MODEL_OP_2(0x80, 0x4e)
+
+#define STATUS_SUCCESS 0x00
+#define STATUS_INVALID_ADDRESS 0x01
+#define STATUS_INVALID_MODEL 0x02
+#define STATUS_INVALID_APPKEY 0x03
+#define STATUS_INVALID_NETKEY 0x04
+#define STATUS_INSUFF_RESOURCES 0x05
+#define STATUS_IDX_ALREADY_STORED 0x06
+#define STATUS_NVAL_PUB_PARAM 0x07
+#define STATUS_NOT_SUB_MOD 0x08
+#define STATUS_STORAGE_FAIL 0x09
+#define STATUS_FEAT_NOT_SUPP 0x0a
+#define STATUS_CANNOT_UPDATE 0x0b
+#define STATUS_CANNOT_REMOVE 0x0c
+#define STATUS_CANNOT_BIND 0x0d
+#define STATUS_TEMP_STATE_CHG_FAIL 0x0e
+#define STATUS_CANNOT_SET 0x0f
+#define STATUS_UNSPECIFIED 0x10
+#define STATUS_INVALID_BINDING 0x11
+
+int bt_mesh_cfg_srv_init(struct bt_mesh_model *model, bool primary);
+int bt_mesh_health_srv_init(struct bt_mesh_model *model, bool primary);
+
+int bt_mesh_cfg_cli_init(struct bt_mesh_model *model, bool primary);
+int bt_mesh_health_cli_init(struct bt_mesh_model *model, bool primary);
+
+void bt_mesh_cfg_reset(void);
+
+void bt_mesh_heartbeat(u16_t src, u16_t dst, u8_t hops, u16_t feat);
+
+void bt_mesh_attention(struct bt_mesh_model *model, u8_t time);
+
+u8_t *bt_mesh_label_uuid_get(u16_t addr);
+
+struct bt_mesh_hb_pub *bt_mesh_hb_pub_get(void);
+struct bt_mesh_cfg_srv *bt_mesh_cfg_get(void);
+
+u8_t bt_mesh_net_transmit_get(void);
+u8_t bt_mesh_relay_get(void);
+u8_t bt_mesh_friend_get(void);
+u8_t bt_mesh_relay_retransmit_get(void);
+u8_t bt_mesh_beacon_get(void);
+u8_t bt_mesh_gatt_proxy_get(void);
+u8_t bt_mesh_default_ttl_get(void);
+
+void bt_mesh_subnet_del(struct bt_mesh_subnet *sub);
+
+struct bt_mesh_app_key *bt_mesh_app_key_alloc(u16_t app_idx);
+void bt_mesh_app_key_del(struct bt_mesh_app_key *key);
+
+
+static inline void key_idx_pack(struct net_buf_simple *buf,
+ u16_t idx1, u16_t idx2)
+{
+ net_buf_simple_add_le16(buf, idx1 | ((idx2 & 0x00f) << 12));
+ net_buf_simple_add_u8(buf, idx2 >> 4);
+}
+
+static inline void key_idx_unpack(struct net_buf_simple *buf,
+ u16_t *idx1, u16_t *idx2)
+{
+ *idx1 = sys_get_le16(&buf->data[0]) & 0xfff;
+ *idx2 = sys_get_le16(&buf->data[1]) >> 4;
+ net_buf_simple_pull(buf, 3);
+}
+
+#endif /* #ifndef _FOUNDATION_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/friend.c b/components/bt/ble_mesh/mesh_core/friend.c
new file mode 100644
index 0000000000..f7f26754a4
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/friend.c
@@ -0,0 +1,1324 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#include
+#include
+#include
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+#include "mesh_util.h"
+#include "mesh_main.h"
+#ifdef CONFIG_BT_MESH_FRIEND
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_FRIEND)
+#include "mesh_trace.h"
+
+#include "crypto.h"
+#include "adv.h"
+#include "mesh.h"
+#include "net.h"
+#include "transport.h"
+#include "access.h"
+#include "foundation.h"
+#include "friend.h"
+
+#define FRIEND_BUF_SIZE (BT_MESH_ADV_DATA_SIZE - BT_MESH_NET_HDR_LEN)
+
+/* We reserve one extra buffer for each friendship, since we need to be able
+ * to resend the last sent PDU, which sits separately outside of the queue.
+ */
+#define FRIEND_BUF_COUNT ((CONFIG_BT_MESH_FRIEND_QUEUE_SIZE + 1) * \
+ CONFIG_BT_MESH_FRIEND_LPN_COUNT)
+
+#define FRIEND_ADV(buf) CONTAINER_OF(BT_MESH_ADV(buf), \
+ struct friend_adv, adv)
+
+/* PDUs from Friend to the LPN should only be transmitted once with the
+ * smallest possible interval (20ms).
+ */
+#define FRIEND_XMIT BT_MESH_TRANSMIT(0, 20)
+
+struct friend_pdu_info {
+ u16_t src;
+ u16_t dst;
+
+ u8_t seq[3];
+
+ u8_t ttl: 7,
+ ctl: 1;
+
+ u32_t iv_index;
+};
+
+NET_BUF_POOL_DEFINE(friend_buf_pool, FRIEND_BUF_COUNT,
+ BT_MESH_ADV_DATA_SIZE, BT_MESH_ADV_USER_DATA_SIZE, NULL);
+
+static struct friend_adv {
+ struct bt_mesh_adv adv;
+ u64_t seq_auth;
+} adv_pool[FRIEND_BUF_COUNT];
+
+static struct bt_mesh_adv *adv_alloc(int id)
+{
+ return &adv_pool[id].adv;
+}
+
+static void discard_buffer(void)
+{
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[0];
+ struct net_buf *buf;
+ int i;
+
+ /* Find the Friend context with the most queued buffers */
+ for (i = 1; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ if (bt_mesh.frnd[i].queue_size > frnd->queue_size) {
+ frnd = &bt_mesh.frnd[i];
+ }
+ }
+
+ buf = net_buf_slist_get(&frnd->queue);
+ __ASSERT_NO_MSG(buf != NULL);
+ BT_WARN("Discarding buffer %p for LPN 0x%04x", buf, frnd->lpn);
+ net_buf_unref(buf);
+}
+
+static struct net_buf *friend_buf_alloc(u16_t src)
+{
+ struct net_buf *buf;
+
+ BT_DBG("src 0x%04x", src);
+
+ do {
+ buf = bt_mesh_adv_create_from_pool(&friend_buf_pool, adv_alloc,
+ BT_MESH_ADV_DATA,
+ BT_MESH_TRANSMIT_COUNT(FRIEND_XMIT),
+ BT_MESH_TRANSMIT_INT(FRIEND_XMIT),
+ K_NO_WAIT);
+ if (!buf) {
+ discard_buffer();
+ }
+ } while (!buf);
+
+ BT_MESH_ADV(buf)->addr = src;
+ FRIEND_ADV(buf)->seq_auth = TRANS_SEQ_AUTH_NVAL;
+
+ BT_DBG("allocated buf %p", buf);
+
+ return buf;
+}
+
+static bool is_lpn_unicast(struct bt_mesh_friend *frnd, u16_t addr)
+{
+ if (frnd->lpn == BT_MESH_ADDR_UNASSIGNED) {
+ return false;
+ }
+
+ return (addr >= frnd->lpn && addr < (frnd->lpn + frnd->num_elem));
+}
+
+struct bt_mesh_friend *bt_mesh_friend_find(u16_t net_idx, u16_t lpn_addr,
+ bool valid, bool established)
+{
+ int i;
+
+ BT_DBG("net_idx 0x%04x lpn_addr 0x%04x", net_idx, lpn_addr);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+
+ if (valid && !frnd->valid) {
+ continue;
+ }
+
+ if (established && !frnd->established) {
+ continue;
+ }
+
+ if (net_idx != BT_MESH_KEY_ANY && frnd->net_idx != net_idx) {
+ continue;
+ }
+
+ if (is_lpn_unicast(frnd, lpn_addr)) {
+ return frnd;
+ }
+ }
+
+ return NULL;
+}
+
+/* Intentionally start a little bit late into the ReceiveWindow when
+ * it's large enough. This may improve reliability with some platforms,
+ * like the PTS, where the receiver might not have sufficiently compensated
+ * for internal latencies required to start scanning.
+ */
+static s32_t recv_delay(struct bt_mesh_friend *frnd)
+{
+#if CONFIG_BT_MESH_FRIEND_RECV_WIN > 50
+ return (s32_t)frnd->recv_delay + (CONFIG_BT_MESH_FRIEND_RECV_WIN / 5);
+#else
+ return frnd->recv_delay;
+#endif
+}
+
+static void friend_clear(struct bt_mesh_friend *frnd)
+{
+ int i;
+
+ BT_DBG("LPN 0x%04x", frnd->lpn);
+
+ k_delayed_work_cancel(&frnd->timer);
+
+ friend_cred_del(frnd->net_idx, frnd->lpn);
+
+ if (frnd->last) {
+ /* Cancel the sending if necessary */
+ if (frnd->pending_buf) {
+ BT_MESH_ADV(frnd->last)->busy = 0;
+ }
+
+ net_buf_unref(frnd->last);
+ frnd->last = NULL;
+ }
+
+ while (!sys_slist_is_empty(&frnd->queue)) {
+ net_buf_unref(net_buf_slist_get(&frnd->queue));
+ }
+
+ for (i = 0; i < ARRAY_SIZE(frnd->seg); i++) {
+ struct bt_mesh_friend_seg *seg = &frnd->seg[i];
+
+ while (!sys_slist_is_empty(&seg->queue)) {
+ net_buf_unref(net_buf_slist_get(&seg->queue));
+ }
+ }
+
+ frnd->valid = 0;
+ frnd->established = 0;
+ frnd->pending_buf = 0;
+ frnd->fsn = 0;
+ frnd->queue_size = 0;
+ frnd->pending_req = 0;
+ memset(frnd->sub_list, 0, sizeof(frnd->sub_list));
+}
+
+void bt_mesh_friend_clear_net_idx(u16_t net_idx)
+{
+ int i;
+
+ BT_DBG("net_idx 0x%04x", net_idx);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+
+ if (frnd->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ if (net_idx == BT_MESH_KEY_ANY || frnd->net_idx == net_idx) {
+ friend_clear(frnd);
+ }
+ }
+}
+
+void bt_mesh_friend_sec_update(u16_t net_idx)
+{
+ int i;
+
+ BT_DBG("net_idx 0x%04x", net_idx);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+
+ if (frnd->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ if (net_idx == BT_MESH_KEY_ANY || frnd->net_idx == net_idx) {
+ frnd->sec_update = 1;
+ }
+ }
+}
+
+int bt_mesh_friend_clear(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_clear *msg = (void *)buf->data;
+ struct bt_mesh_friend *frnd;
+ u16_t lpn_addr, lpn_counter;
+ struct bt_mesh_net_tx tx = {
+ .sub = rx->sub,
+ .ctx = &rx->ctx,
+ .src = bt_mesh_primary_addr(),
+ .xmit = bt_mesh_net_transmit_get(),
+ };
+ struct bt_mesh_ctl_friend_clear_confirm cfm;
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Clear");
+ return -EINVAL;
+ }
+
+ lpn_addr = sys_be16_to_cpu(msg->lpn_addr);
+ lpn_counter = sys_be16_to_cpu(msg->lpn_counter);
+
+ BT_DBG("LPN addr 0x%04x counter 0x%04x", lpn_addr, lpn_counter);
+
+ frnd = bt_mesh_friend_find(rx->sub->net_idx, lpn_addr, false, false);
+ if (!frnd) {
+ BT_WARN("No matching LPN addr 0x%04x", lpn_addr);
+ return 0;
+ }
+
+ /* A Friend Clear message is considered valid if the result of the
+ * subtraction of the value of the LPNCounter field of the Friend
+ * Request message (the one that initiated the friendship) from the
+ * value of the LPNCounter field of the Friend Clear message, modulo
+ * 65536, is in the range 0 to 255 inclusive.
+ */
+ if (lpn_counter - frnd->lpn_counter > 255) {
+ BT_WARN("LPN Counter out of range (old %u new %u)",
+ frnd->lpn_counter, lpn_counter);
+ return 0;
+ }
+
+ tx.ctx->send_ttl = BT_MESH_TTL_MAX;
+
+ cfm.lpn_addr = msg->lpn_addr;
+ cfm.lpn_counter = msg->lpn_counter;
+
+ bt_mesh_ctl_send(&tx, TRANS_CTL_OP_FRIEND_CLEAR_CFM, &cfm,
+ sizeof(cfm), NULL, NULL, NULL);
+
+ friend_clear(frnd);
+
+ return 0;
+}
+
+static void friend_sub_add(struct bt_mesh_friend *frnd, u16_t addr)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(frnd->sub_list); i++) {
+ if (frnd->sub_list[i] == BT_MESH_ADDR_UNASSIGNED) {
+ frnd->sub_list[i] = addr;
+ return;
+ }
+ }
+
+ BT_WARN("No space in friend subscription list");
+}
+
+static void friend_sub_rem(struct bt_mesh_friend *frnd, u16_t addr)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(frnd->sub_list); i++) {
+ if (frnd->sub_list[i] == addr) {
+ frnd->sub_list[i] = BT_MESH_ADDR_UNASSIGNED;
+ return;
+ }
+ }
+}
+
+static struct net_buf *create_friend_pdu(struct bt_mesh_friend *frnd,
+ struct friend_pdu_info *info,
+ struct net_buf_simple *sdu)
+{
+ struct bt_mesh_subnet *sub;
+ const u8_t *enc, *priv;
+ struct net_buf *buf;
+ u8_t nid;
+
+ sub = bt_mesh_subnet_get(frnd->net_idx);
+ __ASSERT_NO_MSG(sub != NULL);
+
+ buf = friend_buf_alloc(info->src);
+
+ /* Friend Offer needs master security credentials */
+ if (info->ctl && TRANS_CTL_OP(sdu->data) == TRANS_CTL_OP_FRIEND_OFFER) {
+ enc = sub->keys[sub->kr_flag].enc;
+ priv = sub->keys[sub->kr_flag].privacy;
+ nid = sub->keys[sub->kr_flag].nid;
+ } else {
+ if (friend_cred_get(sub, frnd->lpn, &nid, &enc, &priv)) {
+ BT_ERR("friend_cred_get failed");
+ goto failed;
+ }
+ }
+
+ net_buf_add_u8(buf, (nid | (info->iv_index & 1) << 7));
+
+ if (info->ctl) {
+ net_buf_add_u8(buf, info->ttl | 0x80);
+ } else {
+ net_buf_add_u8(buf, info->ttl);
+ }
+
+ net_buf_add_mem(buf, info->seq, sizeof(info->seq));
+
+ net_buf_add_be16(buf, info->src);
+ net_buf_add_be16(buf, info->dst);
+
+ net_buf_add_mem(buf, sdu->data, sdu->len);
+
+ /* We re-encrypt and obfuscate using the received IVI rather than
+ * the normal TX IVI (which may be different) since the transport
+ * layer nonce includes the IVI.
+ */
+ if (bt_mesh_net_encrypt(enc, &buf->b, info->iv_index, false)) {
+ BT_ERR("Re-encrypting failed");
+ goto failed;
+ }
+
+ if (bt_mesh_net_obfuscate(buf->data, info->iv_index, priv)) {
+ BT_ERR("Re-obfuscating failed");
+ goto failed;
+ }
+
+ return buf;
+
+failed:
+ net_buf_unref(buf);
+ return NULL;
+}
+
+static struct net_buf *encode_friend_ctl(struct bt_mesh_friend *frnd,
+ u8_t ctl_op,
+ struct net_buf_simple *sdu)
+{
+ struct friend_pdu_info info;
+ u32_t seq;
+
+ BT_DBG("LPN 0x%04x", frnd->lpn);
+
+ net_buf_simple_push_u8(sdu, TRANS_CTL_HDR(ctl_op, 0));
+
+ info.src = bt_mesh_primary_addr();
+ info.dst = frnd->lpn;
+
+ info.ctl = 1;
+ info.ttl = 0;
+
+ seq = bt_mesh_next_seq();
+ info.seq[0] = seq >> 16;
+ info.seq[1] = seq >> 8;
+ info.seq[2] = seq;
+
+ info.iv_index = BT_MESH_NET_IVI_TX;
+
+ return create_friend_pdu(frnd, &info, sdu);
+}
+
+static struct net_buf *encode_update(struct bt_mesh_friend *frnd, u8_t md)
+{
+ struct bt_mesh_ctl_friend_update *upd;
+ struct net_buf_simple *sdu = NET_BUF_SIMPLE(1 + sizeof(*upd));
+ struct bt_mesh_subnet *sub = bt_mesh_subnet_get(frnd->net_idx);
+
+ __ASSERT_NO_MSG(sub != NULL);
+
+ BT_DBG("lpn 0x%04x md 0x%02x", frnd->lpn, md);
+
+ net_buf_simple_init(sdu, 1);
+
+ upd = net_buf_simple_add(sdu, sizeof(*upd));
+ upd->flags = bt_mesh_net_flags(sub);
+ upd->iv_index = sys_cpu_to_be32(bt_mesh.iv_index);
+ upd->md = md;
+
+ return encode_friend_ctl(frnd, TRANS_CTL_OP_FRIEND_UPDATE, sdu);
+}
+
+static void enqueue_sub_cfm(struct bt_mesh_friend *frnd, u8_t xact)
+{
+ struct bt_mesh_ctl_friend_sub_confirm *cfm;
+ struct net_buf_simple *sdu = NET_BUF_SIMPLE(1 + sizeof(*cfm));
+ struct net_buf *buf;
+
+ BT_DBG("lpn 0x%04x xact 0x%02x", frnd->lpn, xact);
+
+ net_buf_simple_init(sdu, 1);
+
+ cfm = net_buf_simple_add(sdu, sizeof(*cfm));
+ cfm->xact = xact;
+
+ buf = encode_friend_ctl(frnd, TRANS_CTL_OP_FRIEND_SUB_CFM, sdu);
+ if (!buf) {
+ BT_ERR("Unable to encode Subscription List Confirmation");
+ return;
+ }
+
+ if (frnd->last) {
+ BT_DBG("Discarding last PDU");
+ net_buf_unref(frnd->last);
+ }
+
+ frnd->last = buf;
+ frnd->send_last = 1;
+}
+
+static void friend_recv_delay(struct bt_mesh_friend *frnd)
+{
+ frnd->pending_req = 1;
+ k_delayed_work_submit(&frnd->timer, recv_delay(frnd));
+ BT_DBG("Waiting RecvDelay of %d ms", recv_delay(frnd));
+}
+
+int bt_mesh_friend_sub_add(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_friend *frnd;
+ u8_t xact;
+
+ if (buf->len < BT_MESH_FRIEND_SUB_MIN_LEN) {
+ BT_WARN("Too short Friend Subscription Add");
+ return -EINVAL;
+ }
+
+ frnd = bt_mesh_friend_find(rx->sub->net_idx, rx->ctx.addr, true, true);
+ if (!frnd) {
+ BT_WARN("No matching LPN addr 0x%04x", rx->ctx.addr);
+ return 0;
+ }
+
+ if (frnd->pending_buf) {
+ BT_WARN("Previous buffer not yet sent!");
+ return 0;
+ }
+
+ friend_recv_delay(frnd);
+
+ xact = net_buf_simple_pull_u8(buf);
+
+ while (buf->len >= 2) {
+ friend_sub_add(frnd, net_buf_simple_pull_be16(buf));
+ }
+
+ enqueue_sub_cfm(frnd, xact);
+
+ return 0;
+}
+
+int bt_mesh_friend_sub_rem(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_friend *frnd;
+ u8_t xact;
+
+ if (buf->len < BT_MESH_FRIEND_SUB_MIN_LEN) {
+ BT_WARN("Too short Friend Subscription Remove");
+ return -EINVAL;
+ }
+
+ frnd = bt_mesh_friend_find(rx->sub->net_idx, rx->ctx.addr, true, true);
+ if (!frnd) {
+ BT_WARN("No matching LPN addr 0x%04x", rx->ctx.addr);
+ return 0;
+ }
+
+ if (frnd->pending_buf) {
+ BT_WARN("Previous buffer not yet sent!");
+ return 0;
+ }
+
+ friend_recv_delay(frnd);
+
+ xact = net_buf_simple_pull_u8(buf);
+
+ while (buf->len >= 2) {
+ friend_sub_rem(frnd, net_buf_simple_pull_be16(buf));
+ }
+
+ enqueue_sub_cfm(frnd, xact);
+
+ return 0;
+}
+
+static void enqueue_buf(struct bt_mesh_friend *frnd, struct net_buf *buf)
+{
+ net_buf_slist_put(&frnd->queue, buf);
+ frnd->queue_size++;
+}
+
+static void enqueue_update(struct bt_mesh_friend *frnd, u8_t md)
+{
+ struct net_buf *buf;
+
+ buf = encode_update(frnd, md);
+ if (!buf) {
+ BT_ERR("Unable to encode Friend Update");
+ return;
+ }
+
+ frnd->sec_update = 0;
+ enqueue_buf(frnd, buf);
+}
+
+int bt_mesh_friend_poll(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_poll *msg = (void *)buf->data;
+ struct bt_mesh_friend *frnd;
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Poll");
+ return -EINVAL;
+ }
+
+ frnd = bt_mesh_friend_find(rx->sub->net_idx, rx->ctx.addr, true, false);
+ if (!frnd) {
+ BT_WARN("No matching LPN addr 0x%04x", rx->ctx.addr);
+ return 0;
+ }
+
+ if (msg->fsn & ~1) {
+ BT_WARN("Prohibited (non-zero) padding bits");
+ return -EINVAL;
+ }
+
+ if (frnd->pending_buf) {
+ BT_WARN("Previous buffer not yet sent");
+ return 0;
+ }
+
+ BT_DBG("msg->fsn %u frnd->fsn %u", (msg->fsn & 1), frnd->fsn);
+
+ friend_recv_delay(frnd);
+
+ if (!frnd->established) {
+ BT_DBG("Friendship established with 0x%04x", frnd->lpn);
+ frnd->established = 1;
+ }
+
+ if (msg->fsn == frnd->fsn && frnd->last) {
+ BT_DBG("Re-sending last PDU");
+ frnd->send_last = 1;
+ } else {
+ if (frnd->last) {
+ net_buf_unref(frnd->last);
+ frnd->last = NULL;
+ }
+
+ frnd->fsn = msg->fsn;
+
+ if (sys_slist_is_empty(&frnd->queue)) {
+ enqueue_update(frnd, 0);
+ BT_DBG("Enqueued Friend Update to empty queue");
+ }
+ }
+
+ return 0;
+}
+
+static struct bt_mesh_friend *find_clear(u16_t prev_friend)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+
+ if (frnd->clear.frnd == prev_friend) {
+ return frnd;
+ }
+ }
+
+ return NULL;
+}
+
+static void friend_clear_sent(int err, void *user_data)
+{
+ struct bt_mesh_friend *frnd = user_data;
+
+ k_delayed_work_submit(&frnd->clear.timer,
+ K_SECONDS(frnd->clear.repeat_sec));
+ frnd->clear.repeat_sec *= 2;
+}
+
+static const struct bt_mesh_send_cb clear_sent_cb = {
+ .end = friend_clear_sent,
+};
+
+static void send_friend_clear(struct bt_mesh_friend *frnd)
+{
+ struct bt_mesh_msg_ctx ctx = {
+ .net_idx = frnd->net_idx,
+ .app_idx = BT_MESH_KEY_UNUSED,
+ .addr = frnd->clear.frnd,
+ .send_ttl = BT_MESH_TTL_MAX,
+ };
+ struct bt_mesh_net_tx tx = {
+ .sub = &bt_mesh.sub[0],
+ .ctx = &ctx,
+ .src = bt_mesh_primary_addr(),
+ .xmit = bt_mesh_net_transmit_get(),
+ };
+ struct bt_mesh_ctl_friend_clear req = {
+ .lpn_addr = sys_cpu_to_be16(frnd->lpn),
+ .lpn_counter = sys_cpu_to_be16(frnd->lpn_counter),
+ };
+
+ BT_DBG("%s", __func__);
+
+ bt_mesh_ctl_send(&tx, TRANS_CTL_OP_FRIEND_CLEAR, &req,
+ sizeof(req), NULL, &clear_sent_cb, frnd);
+}
+
+static void clear_timeout(struct k_work *work)
+{
+ struct bt_mesh_friend *frnd = CONTAINER_OF(work, struct bt_mesh_friend,
+ clear.timer.work);
+ u32_t duration;
+
+ BT_DBG("LPN 0x%04x (old) Friend 0x%04x", frnd->lpn, frnd->clear.frnd);
+
+ duration = k_uptime_get_32() - frnd->clear.start;
+ if (duration > 2 * frnd->poll_to) {
+ BT_DBG("Clear Procedure timer expired");
+ frnd->clear.frnd = BT_MESH_ADDR_UNASSIGNED;
+ return;
+ }
+
+ send_friend_clear(frnd);
+}
+
+static void clear_procedure_start(struct bt_mesh_friend *frnd)
+{
+ BT_DBG("LPN 0x%04x (old) Friend 0x%04x", frnd->lpn, frnd->clear.frnd);
+
+ frnd->clear.start = k_uptime_get_32() + (2 * frnd->poll_to);
+ frnd->clear.repeat_sec = 1;
+
+ send_friend_clear(frnd);
+}
+
+int bt_mesh_friend_clear_cfm(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_clear_confirm *msg = (void *)buf->data;
+ struct bt_mesh_friend *frnd;
+ u16_t lpn_addr, lpn_counter;
+
+ BT_DBG("%s", __func__);
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Clear Confirm");
+ return -EINVAL;
+ }
+
+ frnd = find_clear(rx->ctx.addr);
+ if (!frnd) {
+ BT_WARN("No pending clear procedure for 0x%02x", rx->ctx.addr);
+ return 0;
+ }
+
+ lpn_addr = sys_be16_to_cpu(msg->lpn_addr);
+ if (lpn_addr != frnd->lpn) {
+ BT_WARN("LPN address mismatch (0x%04x != 0x%04x)",
+ lpn_addr, frnd->lpn);
+ return 0;
+ }
+
+ lpn_counter = sys_be16_to_cpu(msg->lpn_counter);
+ if (lpn_counter != frnd->lpn_counter) {
+ BT_WARN("LPN counter mismatch (0x%04x != 0x%04x)",
+ lpn_counter, frnd->lpn_counter);
+ return 0;
+ }
+
+ k_delayed_work_cancel(&frnd->clear.timer);
+ frnd->clear.frnd = BT_MESH_ADDR_UNASSIGNED;
+
+ return 0;
+}
+
+static void enqueue_offer(struct bt_mesh_friend *frnd, s8_t rssi)
+{
+ struct bt_mesh_ctl_friend_offer *off;
+ struct net_buf_simple *sdu = NET_BUF_SIMPLE(1 + sizeof(*off));
+ struct net_buf *buf;
+
+ BT_DBG("%s", __func__);
+
+ net_buf_simple_init(sdu, 1);
+
+ off = net_buf_simple_add(sdu, sizeof(*off));
+
+ off->recv_win = CONFIG_BT_MESH_FRIEND_RECV_WIN,
+ off->queue_size = CONFIG_BT_MESH_FRIEND_QUEUE_SIZE,
+ off->sub_list_size = ARRAY_SIZE(frnd->sub_list),
+ off->rssi = rssi,
+ off->frnd_counter = sys_cpu_to_be16(frnd->counter);
+
+ buf = encode_friend_ctl(frnd, TRANS_CTL_OP_FRIEND_OFFER, sdu);
+ if (!buf) {
+ BT_ERR("Unable to encode Friend Offer");
+ return;
+ }
+
+ frnd->counter++;
+
+ if (frnd->last) {
+ net_buf_unref(frnd->last);
+ }
+
+ frnd->last = buf;
+ frnd->send_last = 1;
+}
+
+#define RECV_WIN CONFIG_BT_MESH_FRIEND_RECV_WIN
+#define RSSI_FACT(crit) (((crit) >> 5) & (u8_t)BIT_MASK(2))
+#define RECV_WIN_FACT(crit) (((crit) >> 3) & (u8_t)BIT_MASK(2))
+#define MIN_QUEUE_SIZE_LOG(crit) ((crit) & (u8_t)BIT_MASK(3))
+#define MIN_QUEUE_SIZE(crit) ((u32_t)BIT(MIN_QUEUE_SIZE_LOG(crit)))
+
+static s32_t offer_delay(struct bt_mesh_friend *frnd, s8_t rssi, u8_t crit)
+{
+ /* Scaling factors. The actual values are 1, 1.5, 2 & 2.5, but we
+ * want to avoid floating-point arithmetic.
+ */
+ static const u8_t fact[] = { 10, 15, 20, 25 };
+ s32_t delay;
+
+ BT_DBG("ReceiveWindowFactor %u ReceiveWindow %u RSSIFactor %u RSSI %d",
+ fact[RECV_WIN_FACT(crit)], RECV_WIN,
+ fact[RSSI_FACT(crit)], rssi);
+
+ /* Delay = ReceiveWindowFactor * ReceiveWindow - RSSIFactor * RSSI */
+ delay = (s32_t)fact[RECV_WIN_FACT(crit)] * RECV_WIN;
+ delay -= (s32_t)fact[RSSI_FACT(crit)] * rssi;
+ delay /= 10;
+
+ BT_DBG("Local Delay calculated as %d ms", delay);
+
+ if (delay < 100) {
+ return K_MSEC(100);
+ }
+
+ return K_MSEC(delay);
+}
+
+int bt_mesh_friend_req(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_req *msg = (void *)buf->data;
+ struct bt_mesh_friend *frnd = NULL;
+ u32_t poll_to;
+ int i;
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Request");
+ return -EINVAL;
+ }
+
+ if (msg->recv_delay <= 0x09) {
+ BT_WARN("Prohibited ReceiveDelay (0x%02x)", msg->recv_delay);
+ return -EINVAL;
+ }
+
+ poll_to = (((u32_t)msg->poll_to[0] << 16) |
+ ((u32_t)msg->poll_to[1] << 8) |
+ ((u32_t)msg->poll_to[2]));
+
+ if (poll_to <= 0x000009 || poll_to >= 0x34bc00) {
+ BT_WARN("Prohibited PollTimeout (0x%06x)", poll_to);
+ return -EINVAL;
+ }
+
+ if (msg->num_elem == 0x00) {
+ BT_WARN("Prohibited NumElements value (0x00)");
+ return -EINVAL;
+ }
+
+ if (!BT_MESH_ADDR_IS_UNICAST(rx->ctx.addr + msg->num_elem - 1)) {
+ BT_WARN("LPN elements stretch outside of unicast range");
+ return -EINVAL;
+ }
+
+ if (!MIN_QUEUE_SIZE_LOG(msg->criteria)) {
+ BT_WARN("Prohibited Minimum Queue Size in Friend Request");
+ return -EINVAL;
+ }
+
+ if (CONFIG_BT_MESH_FRIEND_QUEUE_SIZE < MIN_QUEUE_SIZE(msg->criteria)) {
+ BT_WARN("We have a too small Friend Queue size (%u < %u)",
+ CONFIG_BT_MESH_FRIEND_QUEUE_SIZE,
+ MIN_QUEUE_SIZE(msg->criteria));
+ return 0;
+ }
+
+ frnd = bt_mesh_friend_find(rx->sub->net_idx, rx->ctx.addr, true, false);
+ if (frnd) {
+ BT_WARN("Existing LPN re-requesting Friendship");
+ friend_clear(frnd);
+ goto init_friend;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ if (!bt_mesh.frnd[i].valid) {
+ frnd = &bt_mesh.frnd[i];
+ frnd->valid = 1;
+ break;
+ }
+ }
+
+ if (!frnd) {
+ BT_WARN("No free Friend contexts for new LPN");
+ return -ENOMEM;
+ }
+
+init_friend:
+ frnd->lpn = rx->ctx.addr;
+ frnd->num_elem = msg->num_elem;
+ frnd->net_idx = rx->sub->net_idx;
+ frnd->recv_delay = msg->recv_delay;
+ frnd->poll_to = poll_to * 100;
+ frnd->lpn_counter = sys_be16_to_cpu(msg->lpn_counter);
+ frnd->clear.frnd = sys_be16_to_cpu(msg->prev_addr);
+
+ BT_DBG("LPN 0x%04x rssi %d recv_delay %u poll_to %ums",
+ frnd->lpn, rx->rssi, frnd->recv_delay, frnd->poll_to);
+
+ if (BT_MESH_ADDR_IS_UNICAST(frnd->clear.frnd) &&
+ !bt_mesh_elem_find(frnd->clear.frnd)) {
+ clear_procedure_start(frnd);
+ }
+
+ k_delayed_work_submit(&frnd->timer,
+ offer_delay(frnd, rx->rssi, msg->criteria));
+
+ friend_cred_create(rx->sub, frnd->lpn, frnd->lpn_counter,
+ frnd->counter);
+
+ enqueue_offer(frnd, rx->rssi);
+
+ return 0;
+}
+
+static struct bt_mesh_friend_seg *get_seg(struct bt_mesh_friend *frnd,
+ u16_t src, u64_t *seq_auth)
+{
+ struct bt_mesh_friend_seg *unassigned = NULL;
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(frnd->seg); i++) {
+ struct bt_mesh_friend_seg *seg = &frnd->seg[i];
+ struct net_buf *buf = (void *)sys_slist_peek_head(&seg->queue);
+
+ if (buf && BT_MESH_ADV(buf)->addr == src &&
+ FRIEND_ADV(buf)->seq_auth == *seq_auth) {
+ return seg;
+ }
+
+ if (!unassigned && !buf) {
+ unassigned = seg;
+ }
+ }
+
+ return unassigned;
+}
+
+static void enqueue_friend_pdu(struct bt_mesh_friend *frnd,
+ enum bt_mesh_friend_pdu_type type,
+ struct net_buf *buf)
+{
+ struct bt_mesh_friend_seg *seg;
+ struct friend_adv *adv;
+
+ BT_DBG("type %u", type);
+
+ if (type == BT_MESH_FRIEND_PDU_SINGLE) {
+ if (frnd->sec_update) {
+ enqueue_update(frnd, 1);
+ }
+
+ enqueue_buf(frnd, buf);
+ return;
+ }
+
+ adv = FRIEND_ADV(buf);
+ seg = get_seg(frnd, BT_MESH_ADV(buf)->addr, &adv->seq_auth);
+ if (!seg) {
+ BT_ERR("No free friend segment RX contexts for 0x%04x",
+ BT_MESH_ADV(buf)->addr);
+ net_buf_unref(buf);
+ return;
+ }
+
+ net_buf_slist_put(&seg->queue, buf);
+
+ if (type == BT_MESH_FRIEND_PDU_COMPLETE) {
+ if (frnd->sec_update) {
+ enqueue_update(frnd, 1);
+ }
+
+ /* Only acks should have a valid SeqAuth in the Friend queue
+ * (otherwise we can't easily detect them there), so clear
+ * the SeqAuth information from the segments before merging.
+ */
+ SYS_SLIST_FOR_EACH_CONTAINER(&seg->queue, buf, node) {
+ FRIEND_ADV(buf)->seq_auth = TRANS_SEQ_AUTH_NVAL;
+ frnd->queue_size++;
+ }
+
+ sys_slist_merge_slist(&frnd->queue, &seg->queue);
+ }
+}
+
+static void buf_send_start(u16_t duration, int err, void *user_data)
+{
+ struct bt_mesh_friend *frnd = user_data;
+
+ BT_DBG("err %d", err);
+
+ frnd->pending_buf = 0;
+
+ /* Friend Offer doesn't follow the re-sending semantics */
+ if (!frnd->established) {
+ net_buf_unref(frnd->last);
+ frnd->last = NULL;
+ }
+}
+
+static void buf_send_end(int err, void *user_data)
+{
+ struct bt_mesh_friend *frnd = user_data;
+
+ BT_DBG("err %d", err);
+
+ if (frnd->pending_req) {
+ BT_WARN("Another request before previous completed sending");
+ return;
+ }
+
+ if (frnd->established) {
+ k_delayed_work_submit(&frnd->timer, frnd->poll_to);
+ BT_DBG("Waiting %u ms for next poll", frnd->poll_to);
+ } else {
+ /* Friend offer timeout is 1 second */
+ k_delayed_work_submit(&frnd->timer, K_SECONDS(1));
+ BT_DBG("Waiting for first poll");
+ }
+}
+
+static void friend_timeout(struct k_work *work)
+{
+ struct bt_mesh_friend *frnd = CONTAINER_OF(work, struct bt_mesh_friend,
+ timer.work);
+ static const struct bt_mesh_send_cb buf_sent_cb = {
+ .start = buf_send_start,
+ .end = buf_send_end,
+ };
+
+ __ASSERT_NO_MSG(frnd->pending_buf == 0);
+
+ BT_DBG("lpn 0x%04x send_last %u last %p", frnd->lpn,
+ frnd->send_last, frnd->last);
+
+ if (frnd->send_last && frnd->last) {
+ BT_DBG("Sending frnd->last %p", frnd->last);
+ frnd->send_last = 0;
+ goto send_last;
+ }
+
+ if (frnd->established && !frnd->pending_req) {
+ BT_WARN("Friendship lost with 0x%04x", frnd->lpn);
+ friend_clear(frnd);
+ return;
+ }
+
+ frnd->last = net_buf_slist_get(&frnd->queue);
+ if (!frnd->last) {
+ BT_WARN("Friendship not established with 0x%04x", frnd->lpn);
+ friend_clear(frnd);
+ return;
+ }
+
+ BT_DBG("Sending buf %p from Friend Queue of LPN 0x%04x",
+ frnd->last, frnd->lpn);
+ frnd->queue_size--;
+
+send_last:
+ frnd->pending_req = 0;
+ frnd->pending_buf = 1;
+ bt_mesh_adv_send(frnd->last, &buf_sent_cb, frnd);
+}
+
+int bt_mesh_friend_init(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+ int j;
+
+ frnd->net_idx = BT_MESH_KEY_UNUSED;
+
+ sys_slist_init(&frnd->queue);
+
+ k_delayed_work_init(&frnd->timer, friend_timeout);
+ k_delayed_work_init(&frnd->clear.timer, clear_timeout);
+
+ for (j = 0; j < ARRAY_SIZE(frnd->seg); j++) {
+ sys_slist_init(&frnd->seg[j].queue);
+ }
+ }
+
+ return 0;
+}
+
+static void friend_purge_old_ack(struct bt_mesh_friend *frnd, u64_t *seq_auth,
+ u16_t src)
+{
+ sys_snode_t *cur, *prev = NULL;
+
+ BT_DBG("SeqAuth %llx src 0x%04x", *seq_auth, src);
+
+ for (cur = sys_slist_peek_head(&frnd->queue);
+ cur != NULL; prev = cur, cur = sys_slist_peek_next(cur)) {
+ struct net_buf *buf = (void *)cur;
+
+ if (BT_MESH_ADV(buf)->addr == src &&
+ FRIEND_ADV(buf)->seq_auth == *seq_auth) {
+ BT_DBG("Removing old ack from Friend Queue");
+
+ sys_slist_remove(&frnd->queue, prev, cur);
+ frnd->queue_size--;
+ /* Make sure old slist entry state doesn't remain */
+ buf->frags = NULL;
+
+ net_buf_unref(buf);
+ break;
+ }
+ }
+}
+
+static void friend_lpn_enqueue_rx(struct bt_mesh_friend *frnd,
+ struct bt_mesh_net_rx *rx,
+ enum bt_mesh_friend_pdu_type type,
+ u64_t *seq_auth, struct net_buf_simple *sbuf)
+{
+ struct friend_pdu_info info;
+ struct net_buf *buf;
+
+ BT_DBG("LPN 0x%04x queue_size %u", frnd->lpn, frnd->queue_size);
+
+ if (type == BT_MESH_FRIEND_PDU_SINGLE && seq_auth) {
+ friend_purge_old_ack(frnd, seq_auth, rx->ctx.addr);
+ }
+
+ info.src = rx->ctx.addr;
+ info.dst = rx->dst;
+
+ if (rx->net_if == BT_MESH_NET_IF_LOCAL) {
+ info.ttl = rx->ctx.recv_ttl;
+ } else {
+ info.ttl = rx->ctx.recv_ttl - 1;
+ }
+
+ info.ctl = rx->ctl;
+
+ info.seq[0] = (rx->seq >> 16);
+ info.seq[1] = (rx->seq >> 8);
+ info.seq[2] = rx->seq;
+
+ info.iv_index = BT_MESH_NET_IVI_RX(rx);
+
+ buf = create_friend_pdu(frnd, &info, sbuf);
+ if (!buf) {
+ BT_ERR("Failed to encode Friend buffer");
+ return;
+ }
+
+ if (seq_auth) {
+ FRIEND_ADV(buf)->seq_auth = *seq_auth;
+ }
+
+ enqueue_friend_pdu(frnd, type, buf);
+
+ BT_DBG("Queued message for LPN 0x%04x, queue_size %u",
+ frnd->lpn, frnd->queue_size);
+}
+
+static void friend_lpn_enqueue_tx(struct bt_mesh_friend *frnd,
+ struct bt_mesh_net_tx *tx,
+ enum bt_mesh_friend_pdu_type type,
+ u64_t *seq_auth, struct net_buf_simple *sbuf)
+{
+ struct friend_pdu_info info;
+ struct net_buf *buf;
+ u32_t seq;
+
+ BT_DBG("LPN 0x%04x", frnd->lpn);
+
+ if (type == BT_MESH_FRIEND_PDU_SINGLE && seq_auth) {
+ friend_purge_old_ack(frnd, seq_auth, tx->src);
+ }
+
+ info.src = tx->src;
+ info.dst = tx->ctx->addr;
+
+ info.ttl = tx->ctx->send_ttl;
+ info.ctl = (tx->ctx->app_idx == BT_MESH_KEY_UNUSED);
+
+ seq = bt_mesh_next_seq();
+ info.seq[0] = seq >> 16;
+ info.seq[1] = seq >> 8;
+ info.seq[2] = seq;
+
+ info.iv_index = BT_MESH_NET_IVI_TX;
+
+ buf = create_friend_pdu(frnd, &info, sbuf);
+ if (!buf) {
+ BT_ERR("Failed to encode Friend buffer");
+ return;
+ }
+
+ if (seq_auth) {
+ FRIEND_ADV(buf)->seq_auth = *seq_auth;
+ }
+
+ enqueue_friend_pdu(frnd, type, buf);
+
+ BT_DBG("Queued message for LPN 0x%04x", frnd->lpn);
+}
+
+static bool friend_lpn_matches(struct bt_mesh_friend *frnd, u16_t net_idx,
+ u16_t addr)
+{
+ int i;
+
+ if (!frnd->established) {
+ return false;
+ }
+
+ if (net_idx != frnd->net_idx) {
+ return false;
+ }
+
+ if (BT_MESH_ADDR_IS_UNICAST(addr)) {
+ return is_lpn_unicast(frnd, addr);
+ }
+
+ for (i = 0; i < ARRAY_SIZE(frnd->sub_list); i++) {
+ if (frnd->sub_list[i] == addr) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+bool bt_mesh_friend_match(u16_t net_idx, u16_t addr)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+
+ if (friend_lpn_matches(frnd, net_idx, addr)) {
+ BT_DBG("LPN 0x%04x matched address 0x%04x",
+ frnd->lpn, addr);
+ return true;
+ }
+ }
+
+ BT_DBG("No matching LPN for address 0x%04x", addr);
+
+ return false;
+}
+
+void bt_mesh_friend_enqueue_rx(struct bt_mesh_net_rx *rx,
+ enum bt_mesh_friend_pdu_type type,
+ u64_t *seq_auth, struct net_buf_simple *sbuf)
+{
+ int i;
+
+ if (!rx->friend_match ||
+ (rx->ctx.recv_ttl <= 1 && rx->net_if != BT_MESH_NET_IF_LOCAL) ||
+ bt_mesh_friend_get() != BT_MESH_FRIEND_ENABLED) {
+ return;
+ }
+
+ BT_DBG("recv_ttl %u net_idx 0x%04x src 0x%04x dst 0x%04x",
+ rx->ctx.recv_ttl, rx->sub->net_idx, rx->ctx.addr, rx->dst);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+
+ if (friend_lpn_matches(frnd, rx->sub->net_idx, rx->dst)) {
+ friend_lpn_enqueue_rx(frnd, rx, type, seq_auth, sbuf);
+ }
+ }
+}
+
+bool bt_mesh_friend_enqueue_tx(struct bt_mesh_net_tx *tx,
+ enum bt_mesh_friend_pdu_type type,
+ u64_t *seq_auth, struct net_buf_simple *sbuf)
+{
+ bool matched = false;
+ int i;
+
+ if (!bt_mesh_friend_match(tx->sub->net_idx, tx->ctx->addr) ||
+ bt_mesh_friend_get() != BT_MESH_FRIEND_ENABLED) {
+ return matched;
+ }
+
+ BT_DBG("net_idx 0x%04x dst 0x%04x src 0x%04x", tx->sub->net_idx,
+ tx->ctx->addr, tx->src);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+
+ if (friend_lpn_matches(frnd, tx->sub->net_idx, tx->ctx->addr)) {
+ friend_lpn_enqueue_tx(frnd, tx, type, seq_auth, sbuf);
+ matched = true;
+ }
+ }
+
+ return matched;
+}
+
+void bt_mesh_friend_clear_incomplete(struct bt_mesh_subnet *sub, u16_t src,
+ u16_t dst, u64_t *seq_auth)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.frnd); i++) {
+ struct bt_mesh_friend *frnd = &bt_mesh.frnd[i];
+ int j;
+
+ if (!friend_lpn_matches(frnd, sub->net_idx, dst)) {
+ continue;
+ }
+
+ for (j = 0; j < ARRAY_SIZE(frnd->seg); j++) {
+ struct bt_mesh_friend_seg *seg = &frnd->seg[j];
+ struct net_buf *buf;
+
+ buf = (void *)sys_slist_peek_head(&seg->queue);
+ if (!buf) {
+ continue;
+ }
+
+ if (BT_MESH_ADV(buf)->addr != src) {
+ continue;
+ }
+
+ if (FRIEND_ADV(buf)->seq_auth != *seq_auth) {
+ continue;
+ }
+
+ BT_WARN("Clearing incomplete segments for 0x%04x", src);
+
+ while (!sys_slist_is_empty(&seg->queue)) {
+ net_buf_unref(net_buf_slist_get(&seg->queue));
+ }
+ }
+ }
+}
+#endif /* CONFIG_BT_MESH_FRIEND */
+#endif /* #if CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/friend.h b/components/bt/ble_mesh/mesh_core/friend.h
new file mode 100644
index 0000000000..946e88af6b
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/friend.h
@@ -0,0 +1,44 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+enum bt_mesh_friend_pdu_type {
+ BT_MESH_FRIEND_PDU_SINGLE,
+ BT_MESH_FRIEND_PDU_PARTIAL,
+ BT_MESH_FRIEND_PDU_COMPLETE,
+};
+
+bool bt_mesh_friend_match(u16_t net_idx, u16_t addr);
+
+struct bt_mesh_friend *bt_mesh_friend_find(u16_t net_idx, u16_t lpn_addr,
+ bool valid, bool established);
+
+void bt_mesh_friend_enqueue_rx(struct bt_mesh_net_rx *rx,
+ enum bt_mesh_friend_pdu_type type,
+ u64_t *seq_auth, struct net_buf_simple *sbuf);
+bool bt_mesh_friend_enqueue_tx(struct bt_mesh_net_tx *tx,
+ enum bt_mesh_friend_pdu_type type,
+ u64_t *seq_auth, struct net_buf_simple *sbuf);
+
+void bt_mesh_friend_clear_incomplete(struct bt_mesh_subnet *sub, u16_t src,
+ u16_t dst, u64_t *seq_auth);
+
+void bt_mesh_friend_sec_update(u16_t net_idx);
+
+void bt_mesh_friend_clear_net_idx(u16_t net_idx);
+
+int bt_mesh_friend_poll(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf);
+int bt_mesh_friend_req(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf);
+int bt_mesh_friend_clear(struct bt_mesh_net_rx *rx, struct net_buf_simple *buf);
+int bt_mesh_friend_clear_cfm(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf);
+int bt_mesh_friend_sub_add(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf);
+int bt_mesh_friend_sub_rem(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf);
+
+int bt_mesh_friend_init(void);
diff --git a/components/bt/ble_mesh/mesh_core/health_cli.c b/components/bt/ble_mesh/mesh_core/health_cli.c
new file mode 100644
index 0000000000..76be86122a
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/health_cli.c
@@ -0,0 +1,466 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+
+#include "mesh_types.h"
+#include "mesh_util.h"
+#include "sdkconfig.h"
+#include "osi/allocator.h"
+
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_MODEL)
+#include "mesh_trace.h"
+#include "health_cli.h"
+#include "foundation.h"
+#include "common.h"
+#include "btc_ble_mesh_health.h"
+
+s32_t health_msg_timeout;
+
+static bt_mesh_health_client_t *health_cli;
+
+static const bt_mesh_client_op_pair_t health_op_pair[] = {
+ { OP_HEALTH_FAULT_GET, OP_HEALTH_FAULT_STATUS },
+ { OP_HEALTH_FAULT_CLEAR, OP_HEALTH_FAULT_STATUS },
+ { OP_HEALTH_FAULT_TEST, OP_HEALTH_FAULT_STATUS },
+ { OP_HEALTH_PERIOD_GET, OP_HEALTH_PERIOD_STATUS },
+ { OP_HEALTH_PERIOD_SET, OP_HEALTH_PERIOD_STATUS },
+ { OP_ATTENTION_GET, OP_ATTENTION_STATUS },
+ { OP_ATTENTION_SET, OP_ATTENTION_STATUS },
+};
+
+static void timeout_handler(struct k_work *work)
+{
+ bt_mesh_health_client_t *client = NULL;
+ health_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+
+ BT_WARN("Receive health status message timeout");
+
+ node = CONTAINER_OF(work, bt_mesh_client_node_t, timer.work);
+ if (!node || !node->ctx.model) {
+ BT_ERR("%s: node parameter is NULL", __func__);
+ return;
+ }
+
+ client = (bt_mesh_health_client_t *)node->ctx.model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (health_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: internal_data is NULL", __func__);
+ return;
+ }
+
+ bt_mesh_callback_health_status_to_btc(node->opcode, 0x03, node->ctx.model,
+ &node->ctx, NULL, 0);
+
+ bt_mesh_client_free_node(&internal->queue, node);
+
+ return;
+}
+
+static void health_client_cancel(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ void *status, size_t len)
+{
+ health_internal_data_t *data = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ struct net_buf_simple buf = {0};
+ u8_t evt_type = 0xFF;
+
+ if (!model || !ctx || !status || !len) {
+ BT_ERR("%s: invalid parameter", __func__);
+ return;
+ }
+
+ data = (health_internal_data_t *)health_cli->internal_data;
+ if (!data) {
+ BT_ERR("%s: health client internal_data is NULL", __func__);
+ return;
+ }
+
+ /* If it is a publish message, sent to the user directly. */
+ buf.data = (u8_t *)status;
+ buf.len = (u16_t)len;
+ node = bt_mesh_is_model_message_publish(model, ctx, &buf, true);
+ if (!node) {
+ BT_DBG("Unexpected health status message 0x%x", ctx->recv_op);
+ } else {
+ switch (node->opcode) {
+ case OP_HEALTH_FAULT_GET:
+ case OP_HEALTH_PERIOD_GET:
+ case OP_ATTENTION_GET:
+ evt_type = 0x00;
+ break;
+ case OP_HEALTH_FAULT_CLEAR:
+ case OP_HEALTH_FAULT_TEST:
+ case OP_HEALTH_PERIOD_SET:
+ case OP_ATTENTION_SET:
+ evt_type = 0x01;
+ break;
+ default:
+ break;
+ }
+
+ bt_mesh_callback_health_status_to_btc(node->opcode, evt_type, model,
+ ctx, (const u8_t *)status, len);
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&data->queue, node);
+ }
+
+ switch (ctx->recv_op) {
+ case OP_HEALTH_FAULT_STATUS: {
+ struct bt_mesh_health_fault_status *val;
+ val = (struct bt_mesh_health_fault_status *)status;
+ bt_mesh_free_buf(val->fault_array);
+ break;
+ }
+ default:
+ break;
+ }
+}
+
+static void health_fault_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_health_fault_status status = {0};
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status.test_id = net_buf_simple_pull_u8(buf);
+ status.cid = net_buf_simple_pull_le16(buf);
+ status.fault_array = bt_mesh_alloc_buf(buf->len);
+ if (!status.fault_array) {
+ BT_ERR("%s: allocate memory for fault_array fail", __func__);
+ return;
+ }
+
+ net_buf_simple_init(status.fault_array, 0);
+ net_buf_simple_add_mem(status.fault_array, buf->data, buf->len);
+
+ health_client_cancel(model, ctx, &status, sizeof(struct bt_mesh_health_fault_status));
+}
+
+static void health_current_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ bt_mesh_client_node_t *node = NULL;
+ u8_t test_id;
+ u16_t cid;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ /* Health current status is a publish message, sent to the user directly. */
+ if (!(node = bt_mesh_is_model_message_publish(model, ctx, buf, true))) {
+ return;
+ }
+
+ test_id = net_buf_simple_pull_u8(buf);
+ cid = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("Test ID 0x%02x Company ID 0x%04x Fault Count %u",
+ test_id, cid, buf->len);
+}
+
+static void health_period_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t status = 0;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status = net_buf_simple_pull_u8(buf);
+
+ health_client_cancel(model, ctx, &status, sizeof(u8_t));
+}
+
+static void health_attention_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t status = 0;
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x src 0x%04x len %u: %s",
+ ctx->net_idx, ctx->app_idx, ctx->addr, buf->len,
+ bt_hex(buf->data, buf->len));
+
+ status = net_buf_simple_pull_u8(buf);
+
+ health_client_cancel(model, ctx, &status, sizeof(u8_t));
+}
+
+const struct bt_mesh_model_op bt_mesh_health_cli_op[] = {
+ { OP_HEALTH_FAULT_STATUS, 3, health_fault_status },
+ { OP_HEALTH_CURRENT_STATUS, 3, health_current_status },
+ { OP_HEALTH_PERIOD_STATUS, 1, health_period_status },
+ { OP_ATTENTION_STATUS, 1, health_attention_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+int bt_mesh_health_attention_get(struct bt_mesh_msg_ctx *ctx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_ATTENTION_GET);
+
+ err = bt_mesh_client_send_msg(health_cli->model, OP_ATTENTION_GET, ctx,
+ msg, timeout_handler, health_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_health_attention_set(struct bt_mesh_msg_ctx *ctx,
+ u8_t attention, bool need_ack)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ u32_t opcode;
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ if (need_ack) {
+ opcode = OP_ATTENTION_SET;
+ } else {
+ opcode = OP_ATTENTION_SET_UNREL;
+ }
+ bt_mesh_model_msg_init(msg, opcode);
+ net_buf_simple_add_u8(msg, attention);
+
+ err = bt_mesh_client_send_msg(health_cli->model, opcode, ctx, msg,
+ timeout_handler, health_msg_timeout,
+ need_ack, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_health_period_get(struct bt_mesh_msg_ctx *ctx)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 0 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEALTH_PERIOD_GET);
+
+ err = bt_mesh_client_send_msg(health_cli->model, OP_HEALTH_PERIOD_GET,
+ ctx, msg, timeout_handler, health_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_health_period_set(struct bt_mesh_msg_ctx *ctx,
+ u8_t divisor, bool need_ack)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ u32_t opcode;
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ if (need_ack) {
+ opcode = OP_HEALTH_PERIOD_SET;
+ } else {
+ opcode = OP_HEALTH_PERIOD_SET_UNREL;
+ }
+ bt_mesh_model_msg_init(msg, opcode);
+ net_buf_simple_add_u8(msg, divisor);
+
+ err = bt_mesh_client_send_msg(health_cli->model, opcode, ctx, msg,
+ timeout_handler, health_msg_timeout,
+ need_ack, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_health_fault_test(struct bt_mesh_msg_ctx *ctx,
+ u16_t cid, u8_t test_id, bool need_ack)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 3 + 4);
+ u32_t opcode;
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ if (need_ack) {
+ opcode = OP_HEALTH_FAULT_TEST;
+ } else {
+ opcode = OP_HEALTH_FAULT_TEST_UNREL;
+ }
+ bt_mesh_model_msg_init(msg, opcode);
+ net_buf_simple_add_u8(msg, test_id);
+ net_buf_simple_add_le16(msg, cid);
+
+ err = bt_mesh_client_send_msg(health_cli->model, opcode, ctx, msg,
+ timeout_handler, health_msg_timeout,
+ need_ack, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_health_fault_clear(struct bt_mesh_msg_ctx *ctx,
+ u16_t cid, bool need_ack)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+ u32_t opcode;
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ if (need_ack) {
+ opcode = OP_HEALTH_FAULT_CLEAR;
+ } else {
+ opcode = OP_HEALTH_FAULT_CLEAR_UNREL;
+ }
+ bt_mesh_model_msg_init(msg, opcode);
+ net_buf_simple_add_le16(msg, cid);
+
+ err = bt_mesh_client_send_msg(health_cli->model, opcode, ctx, msg,
+ timeout_handler, health_msg_timeout,
+ need_ack, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+int bt_mesh_health_fault_get(struct bt_mesh_msg_ctx *ctx, u16_t cid)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 2 + 4);
+ int err;
+
+ if (!ctx || !ctx->addr) {
+ return -EINVAL;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEALTH_FAULT_GET);
+ net_buf_simple_add_le16(msg, cid);
+
+ err = bt_mesh_client_send_msg(health_cli->model, OP_HEALTH_FAULT_GET, ctx,
+ msg, timeout_handler, health_msg_timeout,
+ true, NULL, NULL);
+ if (err) {
+ BT_ERR("%s: send failed (err %d)", __func__, err);
+ }
+
+ return err;
+}
+
+s32_t bt_mesh_health_cli_timeout_get(void)
+{
+ return health_msg_timeout;
+}
+
+void bt_mesh_health_cli_timeout_set(s32_t timeout)
+{
+ health_msg_timeout = timeout;
+}
+
+int bt_mesh_health_cli_set(struct bt_mesh_model *model)
+{
+ if (!model || !model->user_data) {
+ BT_ERR("No Health Client context for given model");
+ return -EINVAL;
+ }
+
+ health_cli = model->user_data;
+
+ return 0;
+}
+
+int bt_mesh_health_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ bt_mesh_health_client_t *client = NULL;
+ health_internal_data_t *internal = NULL;
+
+ BT_DBG("primary %u", primary);
+
+ if (!model) {
+ BT_ERR("Health Client model is NULL");
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_health_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("No Health Client context provided");
+ return -EINVAL;
+ }
+
+ /* TODO: call osi_free() when deinit function is invoked*/
+ internal = osi_calloc(sizeof(health_internal_data_t));
+ if (!internal) {
+ BT_ERR("Allocate memory for Health Client internal data fail");
+ return -ENOMEM;
+ }
+
+ sys_slist_init(&internal->queue);
+
+ client->model = model;
+ client->op_pair_size = ARRAY_SIZE(health_op_pair);
+ client->op_pair = health_op_pair;
+ client->internal_data = internal;
+
+ /* Set the default health client pointer */
+ if (!health_cli) {
+ health_cli = client;
+ }
+
+ return 0;
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/health_srv.c b/components/bt/ble_mesh/mesh_core/health_srv.c
new file mode 100644
index 0000000000..ecfad39b1f
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/health_srv.c
@@ -0,0 +1,512 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+#include "mesh_types.h"
+#include "mesh_util.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_MODEL)
+#include "mesh_trace.h"
+
+#include "health_srv.h"
+#include "mesh.h"
+#include "adv.h"
+#include "net.h"
+#include "transport.h"
+#include "access.h"
+#include "foundation.h"
+#include "common.h"
+
+#define HEALTH_TEST_STANDARD 0x00
+
+/* Maximum message length is 384 in BLE Mesh. Here for health fault status,
+ * due to 1 octet opcode and 4 octets TransMIC, 379 octets can be used to
+ * store health fault status.
+ */
+#define HEALTH_FAULT_MAX_LEN 379
+
+/* Health Server context of the primary element */
+struct bt_mesh_health_srv *health_srv;
+
+static void health_get_registered(struct bt_mesh_model *mod,
+ u16_t company_id,
+ struct net_buf_simple *msg)
+{
+ struct bt_mesh_health_srv *srv = mod->user_data;
+ u8_t *test_id;
+
+ BT_DBG("Company ID 0x%04x", company_id);
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEALTH_FAULT_STATUS);
+
+ test_id = net_buf_simple_add(msg, 1);
+ net_buf_simple_add_le16(msg, company_id);
+
+ if (srv->cb && srv->cb->fault_get_reg) {
+ u8_t fault_count = net_buf_simple_tailroom(msg) - 4;
+ int err;
+
+ err = srv->cb->fault_get_reg(mod, company_id, test_id,
+ net_buf_simple_tail(msg),
+ &fault_count);
+ if (err) {
+ BT_ERR("Failed to get faults (err %d)", err);
+ *test_id = HEALTH_TEST_STANDARD;
+ } else {
+ net_buf_simple_add(msg, fault_count);
+ }
+ } else {
+ BT_WARN("No callback for getting faults");
+ *test_id = HEALTH_TEST_STANDARD;
+ }
+}
+
+static size_t health_get_current(struct bt_mesh_model *mod,
+ struct net_buf_simple *msg)
+{
+ struct bt_mesh_health_srv *srv = mod->user_data;
+ const struct bt_mesh_comp *comp;
+ u8_t *test_id, *company_ptr;
+ u16_t company_id;
+ u8_t fault_count;
+ int err;
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return 0;
+ }
+
+ bt_mesh_model_msg_init(msg, OP_HEALTH_CURRENT_STATUS);
+
+ test_id = net_buf_simple_add(msg, 1);
+ company_ptr = net_buf_simple_add(msg, sizeof(company_id));
+ comp = bt_mesh_comp_get();
+
+ if (srv->cb && srv->cb->fault_get_cur) {
+ fault_count = net_buf_simple_tailroom(msg);
+ err = srv->cb->fault_get_cur(mod, test_id, &company_id,
+ net_buf_simple_tail(msg),
+ &fault_count);
+ if (err) {
+ BT_ERR("Failed to get faults (err %d)", err);
+ sys_put_le16(comp->cid, company_ptr);
+ *test_id = HEALTH_TEST_STANDARD;
+ fault_count = 0;
+ } else {
+ sys_put_le16(company_id, company_ptr);
+ net_buf_simple_add(msg, fault_count);
+ }
+ } else {
+ BT_WARN("No callback for getting faults");
+ sys_put_le16(comp->cid, company_ptr);
+ *test_id = HEALTH_TEST_STANDARD;
+ fault_count = 0;
+ }
+
+ return fault_count;
+}
+
+static void health_fault_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *sdu = NULL;
+ u16_t company_id;
+
+ company_id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("company_id 0x%04x", company_id);
+
+ sdu = bt_mesh_alloc_buf(min(BT_MESH_TX_SDU_MAX, HEALTH_FAULT_MAX_LEN));
+ if (!sdu) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return;
+ }
+
+ health_get_registered(model, company_id, sdu);
+
+ if (bt_mesh_model_send(model, ctx, sdu, NULL, NULL)) {
+ BT_ERR("Unable to send Health Current Status response");
+ }
+
+ bt_mesh_free_buf(sdu);
+ return;
+}
+
+static void health_fault_clear_unrel(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_health_srv *srv = model->user_data;
+ u16_t company_id;
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return;
+ }
+
+ company_id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("company_id 0x%04x", company_id);
+
+ if (srv->cb && srv->cb->fault_clear) {
+ srv->cb->fault_clear(model, company_id);
+ }
+}
+
+static void health_fault_clear(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_health_srv *srv = model->user_data;
+ struct net_buf_simple *sdu = NULL;
+ u16_t company_id;
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return;
+ }
+
+ company_id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("company_id 0x%04x", company_id);
+
+ if (srv->cb && srv->cb->fault_clear) {
+ srv->cb->fault_clear(model, company_id);
+ }
+
+ sdu = bt_mesh_alloc_buf(min(BT_MESH_TX_SDU_MAX, HEALTH_FAULT_MAX_LEN));
+ if (!sdu) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return;
+ }
+
+ health_get_registered(model, company_id, sdu);
+
+ if (bt_mesh_model_send(model, ctx, sdu, NULL, NULL)) {
+ BT_ERR("Unable to send Health Current Status response");
+ }
+
+ bt_mesh_free_buf(sdu);
+ return;
+}
+
+static void health_fault_test_unrel(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_health_srv *srv = model->user_data;
+ u16_t company_id;
+ u8_t test_id;
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return;
+ }
+
+ test_id = net_buf_simple_pull_u8(buf);
+ company_id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("test 0x%02x company 0x%04x", test_id, company_id);
+
+ if (srv->cb && srv->cb->fault_test) {
+ srv->cb->fault_test(model, test_id, company_id);
+ }
+}
+
+static void health_fault_test(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_health_srv *srv = model->user_data;
+ struct net_buf_simple *sdu = NULL;
+ u16_t company_id;
+ u8_t test_id;
+
+ BT_DBG("%s", __func__);
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return;
+ }
+
+ test_id = net_buf_simple_pull_u8(buf);
+ company_id = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("test 0x%02x company 0x%04x", test_id, company_id);
+
+ if (srv->cb && srv->cb->fault_test) {
+ int err;
+
+ err = srv->cb->fault_test(model, test_id, company_id);
+ if (err) {
+ BT_WARN("Running fault test failed with err %d", err);
+ return;
+ }
+ }
+
+ sdu = bt_mesh_alloc_buf(min(BT_MESH_TX_SDU_MAX, HEALTH_FAULT_MAX_LEN));
+ if (!sdu) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return;
+ }
+
+ health_get_registered(model, company_id, sdu);
+
+ if (bt_mesh_model_send(model, ctx, sdu, NULL, NULL)) {
+ BT_ERR("Unable to send Health Current Status response");
+ }
+
+ bt_mesh_free_buf(sdu);
+ return;
+}
+
+static void send_attention_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+ struct bt_mesh_health_srv *srv = model->user_data;
+ u8_t time;
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return;
+ }
+
+ time = k_delayed_work_remaining_get(&srv->attn_timer) / 1000;
+ BT_DBG("%u second%s", time, (time == 1) ? "" : "s");
+
+ bt_mesh_model_msg_init(msg, OP_ATTENTION_STATUS);
+
+ net_buf_simple_add_u8(msg, time);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Attention Status");
+ }
+}
+
+static void attention_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("%s", __func__);
+
+ send_attention_status(model, ctx);
+}
+
+static void attention_set_unrel(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t time;
+
+ time = net_buf_simple_pull_u8(buf);
+
+ BT_DBG("%u second%s", time, (time == 1) ? "" : "s");
+
+ bt_mesh_attention(model, time);
+}
+
+static void attention_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("%s", __func__);
+
+ attention_set_unrel(model, ctx, buf);
+
+ send_attention_status(model, ctx);
+}
+
+static void send_health_period_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx)
+{
+ /* Needed size: opcode (2 bytes) + msg + MIC */
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(2 + 1 + 4);
+
+ bt_mesh_model_msg_init(msg, OP_HEALTH_PERIOD_STATUS);
+
+ net_buf_simple_add_u8(msg, model->pub->period_div);
+
+ if (bt_mesh_model_send(model, ctx, msg, NULL, NULL)) {
+ BT_ERR("Unable to send Health Period Status");
+ }
+}
+
+static void health_period_get(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("%s", __func__);
+
+ send_health_period_status(model, ctx);
+}
+
+static void health_period_set_unrel(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ u8_t period;
+
+ period = net_buf_simple_pull_u8(buf);
+ if (period > 15) {
+ BT_WARN("Prohibited period value %u", period);
+ return;
+ }
+
+ BT_DBG("period %u", period);
+
+ model->pub->period_div = period;
+}
+
+static void health_period_set(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("%s", __func__);
+
+ health_period_set_unrel(model, ctx, buf);
+
+ send_health_period_status(model, ctx);
+}
+
+const struct bt_mesh_model_op bt_mesh_health_srv_op[] = {
+ { OP_HEALTH_FAULT_GET, 2, health_fault_get },
+ { OP_HEALTH_FAULT_CLEAR, 2, health_fault_clear },
+ { OP_HEALTH_FAULT_CLEAR_UNREL, 2, health_fault_clear_unrel },
+ { OP_HEALTH_FAULT_TEST, 3, health_fault_test },
+ { OP_HEALTH_FAULT_TEST_UNREL, 3, health_fault_test_unrel },
+ { OP_HEALTH_PERIOD_GET, 0, health_period_get },
+ { OP_HEALTH_PERIOD_SET, 1, health_period_set },
+ { OP_HEALTH_PERIOD_SET_UNREL, 1, health_period_set_unrel },
+ { OP_ATTENTION_GET, 0, attention_get },
+ { OP_ATTENTION_SET, 1, attention_set },
+ { OP_ATTENTION_SET_UNREL, 1, attention_set_unrel },
+ BT_MESH_MODEL_OP_END,
+};
+
+static int health_pub_update(struct bt_mesh_model *mod)
+{
+ struct bt_mesh_model_pub *pub = mod->pub;
+ size_t count;
+
+ BT_DBG("%s", __func__);
+
+ count = health_get_current(mod, pub->msg);
+ if (!count) {
+ pub->period_div = 0;
+ }
+
+ return 0;
+}
+
+int bt_mesh_fault_update(struct bt_mesh_elem *elem)
+{
+ struct bt_mesh_model *mod;
+
+ mod = bt_mesh_model_find(elem, BT_MESH_MODEL_ID_HEALTH_SRV);
+ if (!mod) {
+ return -EINVAL;
+ }
+
+ return bt_mesh_model_publish(mod);
+}
+
+static void attention_off(struct k_work *work)
+{
+ struct bt_mesh_health_srv *srv = CONTAINER_OF(work,
+ struct bt_mesh_health_srv,
+ attn_timer.work);
+ BT_DBG("%s", __func__);
+
+ if (!srv) {
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return;
+ }
+
+ if (srv->cb && srv->cb->attn_off) {
+ srv->cb->attn_off(srv->model);
+ }
+}
+
+int bt_mesh_health_srv_init(struct bt_mesh_model *model, bool primary)
+{
+ struct bt_mesh_health_srv *srv = model->user_data;
+
+ if (!srv) {
+ if (!primary) {
+ return 0;
+ }
+ BT_ERR("%s, No Health Server context provided", __func__);
+ return -EINVAL;
+ }
+
+ if (!model->pub) {
+ BT_ERR("%s, Health Server has no publication support", __func__);
+ return -EINVAL;
+ }
+
+ model->pub->update = health_pub_update;
+
+ k_delayed_work_init(&srv->attn_timer, attention_off);
+
+ srv->model = model;
+
+ if (primary) {
+ health_srv = srv;
+ }
+
+ return 0;
+}
+
+void bt_mesh_attention(struct bt_mesh_model *model, u8_t time)
+{
+ struct bt_mesh_health_srv *srv;
+
+ if (!model) {
+ srv = health_srv;
+ if (!srv) {
+ BT_WARN("%s, No Health Server context provided", __func__);
+ return;
+ }
+ model = srv->model;
+ } else {
+ srv = model->user_data;
+ if (!srv) {
+ BT_WARN("%s, No Health Server context provided", __func__);
+ return;
+ }
+ }
+
+ if (time) {
+ if (srv->cb && srv->cb->attn_on) {
+ srv->cb->attn_on(model);
+ }
+
+ k_delayed_work_submit(&srv->attn_timer, time * 1000);
+ } else {
+ k_delayed_work_cancel(&srv->attn_timer);
+
+ if (srv->cb && srv->cb->attn_off) {
+ srv->cb->attn_off(model);
+ }
+ }
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/include/base64.h b/components/bt/ble_mesh/mesh_core/include/base64.h
new file mode 100644
index 0000000000..76ded32875
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/base64.h
@@ -0,0 +1,79 @@
+/*
+ * RFC 1521 base64 encoding/decoding
+ *
+ * Copyright (C) 2018, Nordic Semiconductor ASA
+ * Copyright (C) 2006-2015, ARM Limited, All Rights Reserved
+ * SPDX-License-Identifier: Apache-2.0
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may
+ * not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ *
+ * Adapted for Zephyr by Carles Cufi (carles.cufi@nordicsemi.no)
+ * - Removed mbedtls_ prefixes
+ * - Reworked coding style
+ */
+#ifndef __BASE64_H__
+#define __BASE64_H__
+
+#include
+#include "mesh_types.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief Encode a buffer into base64 format
+ *
+ * @param dst destination buffer
+ * @param dlen size of the destination buffer
+ * @param olen number of bytes written
+ * @param src source buffer
+ * @param slen amount of data to be encoded
+ *
+ * @return 0 if successful, or -ENOMEM if the buffer is too small.
+ * *olen is always updated to reflect the amount
+ * of data that has (or would have) been written.
+ * If that length cannot be represented, then no data is
+ * written to the buffer and *olen is set to the maximum
+ * length representable as a size_t.
+ *
+ * @note Call this function with dlen = 0 to obtain the
+ * required buffer size in *olen
+ */
+int base64_encode(u8_t *dst, size_t dlen, size_t *olen, const u8_t *src,
+ size_t slen);
+
+/**
+ * @brief Decode a base64-formatted buffer
+ *
+ * @param dst destination buffer (can be NULL for checking size)
+ * @param dlen size of the destination buffer
+ * @param olen number of bytes written
+ * @param src source buffer
+ * @param slen amount of data to be decoded
+ *
+ * @return 0 if successful, -ENOMEM, or -EINVAL if the input data is
+ * not correct. *olen is always updated to reflect the amount
+ * of data that has (or would have) been written.
+ *
+ * @note Call this function with *dst = NULL or dlen = 0 to obtain
+ * the required buffer size in *olen
+ */
+int base64_decode(u8_t *dst, size_t dlen, size_t *olen, const u8_t *src,
+ size_t slen);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __BASE64_H__ */
diff --git a/components/bt/ble_mesh/mesh_core/include/cfg_cli.h b/components/bt/ble_mesh/mesh_core/include/cfg_cli.h
new file mode 100644
index 0000000000..fd31837c54
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/cfg_cli.h
@@ -0,0 +1,297 @@
+/** @file
+ * @brief Bluetooth Mesh Configuration Client Model APIs.
+ */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __BT_MESH_CFG_CLI_H
+#define __BT_MESH_CFG_CLI_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+#include "bt_mesh_client_common.h"
+
+/**
+ * @brief Bluetooth Mesh
+ * @defgroup bt_mesh_cfg_cli Bluetooth Mesh Configuration Client Model
+ * @ingroup bt_mesh
+ * @{
+ */
+
+/* Config client model common structure */
+typedef bt_mesh_client_common_t bt_mesh_config_client_t;
+typedef bt_mesh_internal_data_t config_internal_data_t;
+
+extern const struct bt_mesh_model_op bt_mesh_cfg_cli_op[];
+
+#define BT_MESH_MODEL_CFG_CLI(cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_CFG_CLI, \
+ bt_mesh_cfg_cli_op, NULL, cli_data)
+
+int bt_mesh_cfg_comp_data_get(struct bt_mesh_msg_ctx *ctx, u8_t page);
+
+int bt_mesh_cfg_beacon_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_beacon_set(struct bt_mesh_msg_ctx *ctx, u8_t val);
+
+int bt_mesh_cfg_ttl_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_ttl_set(struct bt_mesh_msg_ctx *ctx, u8_t val);
+
+int bt_mesh_cfg_friend_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_friend_set(struct bt_mesh_msg_ctx *ctx, u8_t val);
+
+int bt_mesh_cfg_gatt_proxy_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_gatt_proxy_set(struct bt_mesh_msg_ctx *ctx, u8_t val);
+
+int bt_mesh_cfg_relay_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_relay_set(struct bt_mesh_msg_ctx *ctx, u8_t new_relay, u8_t new_transmit);
+
+int bt_mesh_cfg_net_key_add(struct bt_mesh_msg_ctx *ctx, u16_t key_net_idx,
+ const u8_t net_key[16]);
+
+int bt_mesh_cfg_app_key_add(struct bt_mesh_msg_ctx *ctx, u16_t key_net_idx,
+ u16_t key_app_idx, const u8_t app_key[16]);
+
+int bt_mesh_cfg_mod_app_bind(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_app_idx, u16_t mod_id, u16_t cid);
+
+struct bt_mesh_cfg_mod_pub {
+ u16_t addr;
+ u16_t app_idx;
+ bool cred_flag;
+ u8_t ttl;
+ u8_t period;
+ u8_t transmit;
+};
+
+int bt_mesh_cfg_mod_pub_get(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_pub_set(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid,
+ struct bt_mesh_cfg_mod_pub *pub);
+
+int bt_mesh_cfg_mod_sub_add(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t sub_addr, u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_sub_del(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t sub_addr, u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_sub_overwrite(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t sub_addr, u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_sub_va_add(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ const u8_t label[16], u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_sub_va_del(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ const u8_t label[16], u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_sub_va_overwrite(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ const u8_t label[16], u16_t mod_id, u16_t cid);
+
+struct bt_mesh_cfg_hb_sub {
+ u16_t src;
+ u16_t dst;
+ u8_t period;
+};
+
+int bt_mesh_cfg_hb_sub_set(struct bt_mesh_msg_ctx *ctx,
+ struct bt_mesh_cfg_hb_sub *sub);
+
+int bt_mesh_cfg_hb_sub_get(struct bt_mesh_msg_ctx *ctx);
+
+struct bt_mesh_cfg_hb_pub {
+ u16_t dst;
+ u8_t count;
+ u8_t period;
+ u8_t ttl;
+ u16_t feat;
+ u16_t net_idx;
+};
+
+int bt_mesh_cfg_hb_pub_set(struct bt_mesh_msg_ctx *ctx,
+ const struct bt_mesh_cfg_hb_pub *pub);
+
+int bt_mesh_cfg_hb_pub_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_node_reset(struct bt_mesh_msg_ctx *ctx);
+
+s32_t bt_mesh_cfg_cli_timeout_get(void);
+void bt_mesh_cfg_cli_timeout_set(s32_t timeout);
+
+/* Configuration Client Status Message Context */
+
+struct bt_mesh_cfg_comp_data_status {
+ u8_t page;
+ struct net_buf_simple *comp_data;
+};
+
+struct bt_mesh_cfg_relay_status {
+ u8_t relay;
+ u8_t retransmit;
+};
+
+struct bt_mesh_cfg_netkey_status {
+ u8_t status;
+ u16_t net_idx;
+};
+
+struct bt_mesh_cfg_appkey_status {
+ u8_t status;
+ u16_t net_idx;
+ u16_t app_idx;
+};
+
+struct bt_mesh_cfg_mod_app_status {
+ u8_t status;
+ u16_t elem_addr;
+ u16_t app_idx;
+ u16_t cid;
+ u16_t mod_id;
+};
+
+struct bt_mesh_cfg_mod_pub_status {
+ u8_t status;
+ u16_t elem_addr;
+ u16_t addr;
+ u16_t app_idx;
+ bool cred_flag;
+ u8_t ttl;
+ u8_t period;
+ u8_t transmit;
+ u16_t cid;
+ u16_t mod_id;
+};
+
+struct bt_mesh_cfg_mod_sub_status {
+ u8_t status;
+ u16_t elem_addr;
+ u16_t sub_addr;
+ u16_t cid;
+ u16_t mod_id;
+};
+
+struct bt_mesh_cfg_hb_sub_status {
+ u8_t status;
+ u16_t src;
+ u16_t dst;
+ u8_t period;
+ u8_t count;
+ u8_t min;
+ u8_t max;
+};
+
+struct bt_mesh_cfg_hb_pub_status {
+ u8_t status;
+ u16_t dst;
+ u8_t count;
+ u8_t period;
+ u8_t ttl;
+ u16_t feat;
+ u16_t net_idx;
+};
+
+struct bt_mesh_cfg_mod_sub_list {
+ u8_t status;
+ u16_t elem_addr;
+ u16_t cid;
+ u16_t mod_id;
+ struct net_buf_simple *addr;
+};
+
+struct bt_mesh_cfg_net_key_list {
+ struct net_buf_simple *net_idx;
+};
+
+struct bt_mesh_cfg_app_key_list {
+ u8_t status;
+ u16_t net_idx;
+ struct net_buf_simple *app_idx;
+};
+
+struct bt_mesh_cfg_node_id_status {
+ u8_t status;
+ u16_t net_idx;
+ u8_t identity;
+};
+
+struct bt_mesh_cfg_mod_app_list {
+ u8_t status;
+ u16_t elem_addr;
+ u16_t cid;
+ u16_t mod_id;
+ struct net_buf_simple *app_idx;
+};
+
+struct bt_mesh_cfg_key_refresh_status {
+ u8_t status;
+ u16_t net_idx;
+ u8_t phase;
+};
+
+struct bt_mesh_cfg_lpn_pollto_status {
+ u16_t lpn_addr;
+ s32_t timeout;
+};
+
+int bt_mesh_cfg_mod_pub_va_set(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid, const u8_t label[16],
+ struct bt_mesh_cfg_mod_pub *pub);
+
+int bt_mesh_cfg_mod_sub_del_all(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_sub_get(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr, u16_t mod_id);
+
+int bt_mesh_cfg_mod_sub_get_vnd(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_net_key_update(struct bt_mesh_msg_ctx *ctx, u16_t net_idx,
+ const u8_t net_key[16]);
+
+int bt_mesh_cfg_net_key_delete(struct bt_mesh_msg_ctx *ctx, u16_t net_idx);
+
+int bt_mesh_cfg_net_key_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_app_key_update(struct bt_mesh_msg_ctx *ctx, u16_t net_idx,
+ u16_t app_idx, const u8_t app_key[16]);
+
+int bt_mesh_cfg_app_key_delete(struct bt_mesh_msg_ctx *ctx, u16_t net_idx, u16_t app_idx);
+
+int bt_mesh_cfg_app_key_get(struct bt_mesh_msg_ctx *ctx, u16_t net_idx);
+
+int bt_mesh_cfg_node_identity_get(struct bt_mesh_msg_ctx *ctx, u16_t net_idx);
+
+int bt_mesh_cfg_node_identity_set(struct bt_mesh_msg_ctx *ctx, u16_t net_idx, u8_t identity);
+
+int bt_mesh_cfg_mod_app_unbind(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t app_idx, u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_mod_app_get(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr, u16_t mod_id);
+
+int bt_mesh_cfg_mod_app_get_vnd(struct bt_mesh_msg_ctx *ctx, u16_t elem_addr,
+ u16_t mod_id, u16_t cid);
+
+int bt_mesh_cfg_kr_phase_get(struct bt_mesh_msg_ctx *ctx, u16_t net_idx);
+
+int bt_mesh_cfg_kr_phase_set(struct bt_mesh_msg_ctx *ctx, u16_t net_idx, u8_t transition);
+
+int bt_mesh_cfg_lpn_timeout_get(struct bt_mesh_msg_ctx *ctx, u16_t lpn_addr);
+
+int bt_mesh_cfg_net_transmit_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_cfg_net_transmit_set(struct bt_mesh_msg_ctx *ctx, u8_t transmit);
+
+/**
+ * @}
+ */
+
+#endif /* __BT_MESH_CFG_CLI_H */
diff --git a/components/bt/ble_mesh/mesh_core/include/cfg_srv.h b/components/bt/ble_mesh/mesh_core/include/cfg_srv.h
new file mode 100644
index 0000000000..633d7a78f0
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/cfg_srv.h
@@ -0,0 +1,72 @@
+/** @file
+ * @brief Bluetooth Mesh Configuration Server Model APIs.
+ */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __BT_MESH_CFG_SRV_H
+#define __BT_MESH_CFG_SRV_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+
+/**
+ * @brief Bluetooth Mesh
+ * @defgroup bt_mesh_cfg_srv Bluetooth Mesh Configuration Server Model
+ * @ingroup bt_mesh
+ * @{
+ */
+
+/** Mesh Configuration Server Model Context */
+struct bt_mesh_cfg_srv {
+ struct bt_mesh_model *model;
+
+ u8_t net_transmit; /* Network Transmit state */
+ u8_t relay; /* Relay Mode state */
+ u8_t relay_retransmit; /* Relay Retransmit state */
+ u8_t beacon; /* Secure Network Beacon state */
+ u8_t gatt_proxy; /* GATT Proxy state */
+ u8_t frnd; /* Friend state */
+ u8_t default_ttl; /* Default TTL */
+
+ /* Heartbeat Publication */
+ struct bt_mesh_hb_pub {
+ struct k_delayed_work timer;
+
+ u16_t dst;
+ u16_t count;
+ u8_t period;
+ u8_t ttl;
+ u16_t feat;
+ u16_t net_idx;
+ } hb_pub;
+
+ /* Heartbeat Subscription */
+ struct bt_mesh_hb_sub {
+ s64_t expiry;
+
+ u16_t src;
+ u16_t dst;
+ u16_t count;
+ u8_t min_hops;
+ u8_t max_hops;
+
+ /* Optional subscription tracking function */
+ void (*func)(u8_t hops, u16_t feat);
+ } hb_sub;
+};
+
+extern const struct bt_mesh_model_op bt_mesh_cfg_srv_op[];
+
+#define BT_MESH_MODEL_CFG_SRV(srv_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_CFG_SRV, \
+ bt_mesh_cfg_srv_op, NULL, srv_data)
+
+/**
+ * @}
+ */
+
+#endif /* __BT_MESH_CFG_SRV_H */
diff --git a/components/bt/ble_mesh/mesh_core/include/dlist.h b/components/bt/ble_mesh/mesh_core/include/dlist.h
new file mode 100644
index 0000000000..11d93ca54c
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/dlist.h
@@ -0,0 +1,495 @@
+/*
+ * Copyright (c) 2013-2015 Wind River Systems, Inc.
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+/**
+ * @file
+ * @brief Doubly-linked list implementation
+ *
+ * Doubly-linked list implementation using inline macros/functions.
+ * This API is not thread safe, and thus if a list is used across threads,
+ * calls to functions must be protected with synchronization primitives.
+ *
+ * The lists are expected to be initialized such that both the head and tail
+ * pointers point to the list itself. Initializing the lists in such a fashion
+ * simplifies the adding and removing of nodes to/from the list.
+ */
+
+#ifndef _misc_dlist__h_
+#define _misc_dlist__h_
+
+#include
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+struct _dnode {
+ union {
+ struct _dnode *head; /* ptr to head of list (sys_dlist_t) */
+ struct _dnode *next; /* ptr to next node (sys_dnode_t) */
+ };
+ union {
+ struct _dnode *tail; /* ptr to tail of list (sys_dlist_t) */
+ struct _dnode *prev; /* ptr to previous node (sys_dnode_t) */
+ };
+};
+
+typedef struct _dnode sys_dlist_t;
+typedef struct _dnode sys_dnode_t;
+
+/**
+ * @brief Provide the primitive to iterate on a list
+ * Note: the loop is unsafe and thus __dn should not be removed
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_NODE(l, n) {
+ *
+ * }
+ *
+ * This and other SYS_DLIST_*() macros are not thread safe.
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __dn A sys_dnode_t pointer to peek each node of the list
+ */
+#define SYS_DLIST_FOR_EACH_NODE(__dl, __dn) \
+ for (__dn = sys_dlist_peek_head(__dl); __dn; \
+ __dn = sys_dlist_peek_next(__dl, __dn))
+
+/**
+ * @brief Provide the primitive to iterate on a list, from a node in the list
+ * Note: the loop is unsafe and thus __dn should not be removed
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_ITERATE_FROM_NODE(l, n) {
+ *
+ * }
+ *
+ * Like SYS_DLIST_FOR_EACH_NODE(), but __dn already contains a node in the list
+ * where to start searching for the next entry from. If NULL, it starts from
+ * the head.
+ *
+ * This and other SYS_DLIST_*() macros are not thread safe.
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __dn A sys_dnode_t pointer to peek each node of the list;
+ * it contains the starting node, or NULL to start from the head
+ */
+#define SYS_DLIST_ITERATE_FROM_NODE(__dl, __dn) \
+ for (__dn = __dn ? sys_dlist_peek_next_no_check(__dl, __dn) \
+ : sys_dlist_peek_head(__dl); \
+ __dn; \
+ __dn = sys_dlist_peek_next(__dl, __dn))
+
+/**
+ * @brief Provide the primitive to safely iterate on a list
+ * Note: __dn can be removed, it will not break the loop.
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_NODE_SAFE(l, n, s) {
+ *
+ * }
+ *
+ * This and other SYS_DLIST_*() macros are not thread safe.
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __dn A sys_dnode_t pointer to peek each node of the list
+ * @param __dns A sys_dnode_t pointer for the loop to run safely
+ */
+#define SYS_DLIST_FOR_EACH_NODE_SAFE(__dl, __dn, __dns) \
+ for (__dn = sys_dlist_peek_head(__dl), \
+ __dns = sys_dlist_peek_next(__dl, __dn); \
+ __dn; __dn = __dns, \
+ __dns = sys_dlist_peek_next(__dl, __dn))
+
+/*
+ * @brief Provide the primitive to resolve the container of a list node
+ * Note: it is safe to use with NULL pointer nodes
+ *
+ * @param __dn A pointer on a sys_dnode_t to get its container
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_CONTAINER(__dn, __cn, __n) \
+ (__dn ? CONTAINER_OF(__dn, __typeof__(*__cn), __n) : NULL)
+/*
+ * @brief Provide the primitive to peek container of the list head
+ *
+ * @param __dl A pointer on a sys_dlist_t to peek
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n) \
+ SYS_DLIST_CONTAINER(sys_dlist_peek_head(__dl), __cn, __n)
+
+/*
+ * @brief Provide the primitive to peek the next container
+ *
+ * @param __dl A pointer on a sys_dlist_t to peek
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n) \
+ ((__cn) ? SYS_DLIST_CONTAINER(sys_dlist_peek_next(__dl, &(__cn->__n)), \
+ __cn, __n) : NULL)
+
+/**
+ * @brief Provide the primitive to iterate on a list under a container
+ * Note: the loop is unsafe and thus __cn should not be detached
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_CONTAINER(l, c, n) {
+ *
+ * }
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __cn A pointer to peek each entry of the list
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_FOR_EACH_CONTAINER(__dl, __cn, __n) \
+ for (__cn = SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n); __cn; \
+ __cn = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n))
+
+/**
+ * @brief Provide the primitive to safely iterate on a list under a container
+ * Note: __cn can be detached, it will not break the loop.
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_DLIST_FOR_EACH_CONTAINER_SAFE(l, c, cn, n) {
+ *
+ * }
+ *
+ * @param __dl A pointer on a sys_dlist_t to iterate on
+ * @param __cn A pointer to peek each entry of the list
+ * @param __cns A pointer for the loop to run safely
+ * @param __n The field name of sys_dnode_t within the container struct
+ */
+#define SYS_DLIST_FOR_EACH_CONTAINER_SAFE(__dl, __cn, __cns, __n) \
+ for (__cn = SYS_DLIST_PEEK_HEAD_CONTAINER(__dl, __cn, __n), \
+ __cns = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n); __cn; \
+ __cn = __cns, \
+ __cns = SYS_DLIST_PEEK_NEXT_CONTAINER(__dl, __cn, __n))
+
+/**
+ * @brief initialize list
+ *
+ * @param list the doubly-linked list
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_init(sys_dlist_t *list)
+{
+ list->head = (sys_dnode_t *)list;
+ list->tail = (sys_dnode_t *)list;
+}
+
+#define SYS_DLIST_STATIC_INIT(ptr_to_list) {{(ptr_to_list)}, {(ptr_to_list)}}
+
+/**
+ * @brief check if a node is the list's head
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node to check
+ *
+ * @return 1 if node is the head, 0 otherwise
+ */
+
+static inline int sys_dlist_is_head(sys_dlist_t *list, sys_dnode_t *node)
+{
+ return list->head == node;
+}
+
+/**
+ * @brief check if a node is the list's tail
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node to check
+ *
+ * @return 1 if node is the tail, 0 otherwise
+ */
+
+static inline int sys_dlist_is_tail(sys_dlist_t *list, sys_dnode_t *node)
+{
+ return list->tail == node;
+}
+
+/**
+ * @brief check if the list is empty
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return 1 if empty, 0 otherwise
+ */
+
+static inline int sys_dlist_is_empty(sys_dlist_t *list)
+{
+ return list->head == list;
+}
+
+/**
+ * @brief check if more than one node present
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return 1 if multiple nodes, 0 otherwise
+ */
+
+static inline int sys_dlist_has_multiple_nodes(sys_dlist_t *list)
+{
+ return list->head != list->tail;
+}
+
+/**
+ * @brief get a reference to the head item in the list
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return a pointer to the head element, NULL if list is empty
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_head(sys_dlist_t *list)
+{
+ return sys_dlist_is_empty(list) ? NULL : list->head;
+}
+
+/**
+ * @brief get a reference to the head item in the list
+ *
+ * The list must be known to be non-empty.
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return a pointer to the head element
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_head_not_empty(sys_dlist_t *list)
+{
+ return list->head;
+}
+
+/**
+ * @brief get a reference to the next item in the list, node is not NULL
+ *
+ * Faster than sys_dlist_peek_next() if node is known not to be NULL.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node from which to get the next element in the list
+ *
+ * @return a pointer to the next element from a node, NULL if node is the tail
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_next_no_check(sys_dlist_t *list,
+ sys_dnode_t *node)
+{
+ return (node == list->tail) ? NULL : node->next;
+}
+
+/**
+ * @brief get a reference to the next item in the list
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the node from which to get the next element in the list
+ *
+ * @return a pointer to the next element from a node, NULL if node is the tail
+ * or NULL (when node comes from reading the head of an empty list).
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_next(sys_dlist_t *list,
+ sys_dnode_t *node)
+{
+ return node ? sys_dlist_peek_next_no_check(list, node) : NULL;
+}
+
+/**
+ * @brief get a reference to the tail item in the list
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return a pointer to the tail element, NULL if list is empty
+ */
+
+static inline sys_dnode_t *sys_dlist_peek_tail(sys_dlist_t *list)
+{
+ return sys_dlist_is_empty(list) ? NULL : list->tail;
+}
+
+/**
+ * @brief add node to tail of list
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the element to append
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_append(sys_dlist_t *list, sys_dnode_t *node)
+{
+ node->next = list;
+ node->prev = list->tail;
+
+ list->tail->next = node;
+ list->tail = node;
+}
+
+/**
+ * @brief add node to head of list
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the element to append
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_prepend(sys_dlist_t *list, sys_dnode_t *node)
+{
+ node->next = list->head;
+ node->prev = list;
+
+ list->head->prev = node;
+ list->head = node;
+}
+
+/**
+ * @brief insert node after a node
+ *
+ * Insert a node after a specified node in a list.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param insert_point the insert point in the list: if NULL, insert at head
+ * @param node the element to append
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_insert_after(sys_dlist_t *list,
+ sys_dnode_t *insert_point, sys_dnode_t *node)
+{
+ if (!insert_point) {
+ sys_dlist_prepend(list, node);
+ } else {
+ node->next = insert_point->next;
+ node->prev = insert_point;
+ insert_point->next->prev = node;
+ insert_point->next = node;
+ }
+}
+
+/**
+ * @brief insert node before a node
+ *
+ * Insert a node before a specified node in a list.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param insert_point the insert point in the list: if NULL, insert at tail
+ * @param node the element to insert
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_insert_before(sys_dlist_t *list,
+ sys_dnode_t *insert_point, sys_dnode_t *node)
+{
+ if (!insert_point) {
+ sys_dlist_append(list, node);
+ } else {
+ node->prev = insert_point->prev;
+ node->next = insert_point;
+ insert_point->prev->next = node;
+ insert_point->prev = node;
+ }
+}
+
+/**
+ * @brief insert node at position
+ *
+ * Insert a node in a location depending on a external condition. The cond()
+ * function checks if the node is to be inserted _before_ the current node
+ * against which it is checked.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ * @param node the element to insert
+ * @param cond a function that determines if the current node is the correct
+ * insert point
+ * @param data parameter to cond()
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_insert_at(sys_dlist_t *list, sys_dnode_t *node,
+ int (*cond)(sys_dnode_t *, void *), void *data)
+{
+ if (sys_dlist_is_empty(list)) {
+ sys_dlist_append(list, node);
+ } else {
+ sys_dnode_t *pos = sys_dlist_peek_head(list);
+
+ while (pos && !cond(pos, data)) {
+ pos = sys_dlist_peek_next(list, pos);
+ }
+ sys_dlist_insert_before(list, pos, node);
+ }
+}
+
+/**
+ * @brief remove a specific node from a list
+ *
+ * The list is implicit from the node. The node must be part of a list.
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param node the node to remove
+ *
+ * @return N/A
+ */
+
+static inline void sys_dlist_remove(sys_dnode_t *node)
+{
+ node->prev->next = node->next;
+ node->next->prev = node->prev;
+}
+
+/**
+ * @brief get the first node in a list
+ *
+ * This and other sys_dlist_*() functions are not thread safe.
+ *
+ * @param list the doubly-linked list to operate on
+ *
+ * @return the first node in the list, NULL if list is empty
+ */
+
+static inline sys_dnode_t *sys_dlist_get(sys_dlist_t *list)
+{
+ sys_dnode_t *node;
+
+ if (sys_dlist_is_empty(list)) {
+ return NULL;
+ }
+
+ node = list->head;
+ sys_dlist_remove(node);
+ return node;
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* _misc_dlist__h_ */
+
diff --git a/components/bt/ble_mesh/mesh_core/include/health_cli.h b/components/bt/ble_mesh/mesh_core/include/health_cli.h
new file mode 100644
index 0000000000..5451343da2
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/health_cli.h
@@ -0,0 +1,78 @@
+/** @file
+ * @brief Bluetooth Mesh Health Client Model APIs.
+ */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __BT_MESH_HEALTH_CLI_H
+#define __BT_MESH_HEALTH_CLI_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+#include "bt_mesh_client_common.h"
+
+/**
+ * @brief Bluetooth Mesh
+ * @defgroup bt_mesh_health_cli Bluetooth Mesh Health Client Model
+ * @ingroup bt_mesh
+ * @{
+ */
+
+/* Health client model common structure */
+typedef bt_mesh_client_common_t bt_mesh_health_client_t;
+typedef bt_mesh_internal_data_t health_internal_data_t;
+
+typedef bt_mesh_internal_data_t health_client_internal_data_t;
+
+extern const struct bt_mesh_model_op bt_mesh_health_cli_op[];
+
+#define BT_MESH_MODEL_HEALTH_CLI(cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_HEALTH_CLI, \
+ bt_mesh_health_cli_op, NULL, cli_data)
+
+int bt_mesh_health_cli_set(struct bt_mesh_model *model);
+
+int bt_mesh_health_fault_get(struct bt_mesh_msg_ctx *ctx, u16_t cid);
+
+int bt_mesh_health_fault_clear(struct bt_mesh_msg_ctx *ctx, u16_t cid,
+ bool need_ack);
+
+int bt_mesh_health_fault_test(struct bt_mesh_msg_ctx *ctx,
+ u16_t cid, u8_t test_id, bool need_ack);
+
+int bt_mesh_health_period_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_health_period_set(struct bt_mesh_msg_ctx *ctx,
+ u8_t divisor, bool need_ack);
+
+int bt_mesh_health_attention_get(struct bt_mesh_msg_ctx *ctx);
+
+int bt_mesh_health_attention_set(struct bt_mesh_msg_ctx *ctx,
+ u8_t attention, bool need_ack);
+
+s32_t bt_mesh_health_cli_timeout_get(void);
+void bt_mesh_health_cli_timeout_set(s32_t timeout);
+
+/* Health Client Status Message Context */
+
+struct bt_mesh_health_current_status {
+ u8_t test_id;
+ u16_t cid;
+ struct net_buf_simple *fault_array;
+};
+
+struct bt_mesh_health_fault_status {
+ u8_t test_id;
+ u16_t cid;
+ struct net_buf_simple *fault_array;
+};
+
+/**
+ * @}
+ */
+
+#endif /* __BT_MESH_HEALTH_CLI_H */
diff --git a/components/bt/ble_mesh/mesh_core/include/health_srv.h b/components/bt/ble_mesh/mesh_core/include/health_srv.h
new file mode 100644
index 0000000000..7e7255c003
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/health_srv.h
@@ -0,0 +1,94 @@
+/** @file
+ * @brief Bluetooth Mesh Health Server Model APIs.
+ */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __BT_MESH_HEALTH_SRV_H
+#define __BT_MESH_HEALTH_SRV_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+
+/**
+ * @brief Bluetooth Mesh Health Server Model
+ * @defgroup bt_mesh_health_srv Bluetooth Mesh Health Server Model
+ * @ingroup bt_mesh
+ * @{
+ */
+
+struct bt_mesh_health_srv_cb {
+ /* Fetch current faults */
+ int (*fault_get_cur)(struct bt_mesh_model *model, u8_t *test_id,
+ u16_t *company_id, u8_t *faults,
+ u8_t *fault_count);
+
+ /* Fetch registered faults */
+ int (*fault_get_reg)(struct bt_mesh_model *model, u16_t company_id,
+ u8_t *test_id, u8_t *faults,
+ u8_t *fault_count);
+
+ /* Clear registered faults */
+ int (*fault_clear)(struct bt_mesh_model *model, u16_t company_id);
+
+ /* Run a specific test */
+ int (*fault_test)(struct bt_mesh_model *model, u8_t test_id,
+ u16_t company_id);
+
+ /* Attention on */
+ void (*attn_on)(struct bt_mesh_model *model);
+
+ /* Attention off */
+ void (*attn_off)(struct bt_mesh_model *model);
+};
+
+/** @def BT_MESH_HEALTH_FAULT_MSG
+ *
+ * A helper to define a health fault message.
+ *
+ * @param max_faults Maximum number of faults the element can have.
+ *
+ * @return a New net_buf_simple of the needed size.
+ */
+#define BT_MESH_HEALTH_FAULT_MSG(max_faults) \
+ NET_BUF_SIMPLE(1 + 3 + (max_faults))
+
+/** Mesh Health Server Model Context */
+struct bt_mesh_health_srv {
+ struct bt_mesh_model *model;
+
+ /* Optional callback struct */
+ const struct bt_mesh_health_srv_cb *cb;
+
+ /* Attention Timer state */
+ struct k_delayed_work attn_timer;
+};
+
+int bt_mesh_fault_update(struct bt_mesh_elem *elem);
+
+extern const struct bt_mesh_model_op bt_mesh_health_srv_op[];
+
+/** @def BT_MESH_MODEL_HEALTH_SRV
+ *
+ * Define a new health server model. Note that this API needs to be
+ * repeated for each element which the application wants to have a
+ * health server model on. Each instance also needs a unique
+ * bt_mesh_health_srv and bt_mesh_model_pub context.
+ *
+ * @param srv Pointer to a unique struct bt_mesh_health_srv.
+ * @param pub Pointer to a unique struct bt_mesh_model_pub.
+ *
+ * @return New mesh model instance.
+ */
+#define BT_MESH_MODEL_HEALTH_SRV(srv, pub) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_HEALTH_SRV, \
+ bt_mesh_health_srv_op, pub, srv)
+
+/**
+ * @}
+ */
+
+#endif /* __BT_MESH_HEALTH_SRV_H */
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_access.h b/components/bt/ble_mesh/mesh_core/include/mesh_access.h
new file mode 100644
index 0000000000..c8353026da
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_access.h
@@ -0,0 +1,449 @@
+/** @file
+ * @brief Bluetooth Mesh Access Layer APIs.
+ */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include "mesh_types.h"
+#include "mesh_util.h"
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+#ifndef __BT_MESH_ACCESS_H
+#define __BT_MESH_ACCESS_H
+
+/**
+ * @brief Bluetooth Mesh Access Layer
+ * @defgroup bt_mesh_access Bluetooth Mesh Access Layer
+ * @ingroup bt_mesh
+ * @{
+ */
+
+#define BT_MESH_ADDR_UNASSIGNED 0x0000
+#define BT_MESH_ADDR_ALL_NODES 0xffff
+#define BT_MESH_ADDR_PROXIES 0xfffc
+#define BT_MESH_ADDR_FRIENDS 0xfffd
+#define BT_MESH_ADDR_RELAYS 0xfffe
+
+#define BT_MESH_KEY_UNUSED 0xffff
+#define BT_MESH_KEY_DEV 0xfffe
+
+/** Helper to define a mesh element within an array.
+ *
+ * In case the element has no SIG or Vendor models the helper
+ * macro BT_MESH_MODEL_NONE can be given instead.
+ *
+ * @param _loc Location Descriptor.
+ * @param _mods Array of models.
+ * @param _vnd_mods Array of vendor models.
+ */
+#define BT_MESH_ELEM(_loc, _mods, _vnd_mods) \
+{ \
+ .loc = (_loc), \
+ .model_count = ARRAY_SIZE(_mods), \
+ .models = (_mods), \
+ .vnd_model_count = ARRAY_SIZE(_vnd_mods), \
+ .vnd_models = (_vnd_mods), \
+}
+
+/** Abstraction that describes a Mesh Element */
+struct bt_mesh_elem {
+ /* Unicast Address. Set at runtime during provisioning. */
+ u16_t addr;
+
+ /* Location Descriptor (GATT Bluetooth Namespace Descriptors) */
+ const u16_t loc;
+
+ const u8_t model_count;
+ const u8_t vnd_model_count;
+
+ struct bt_mesh_model *const models;
+ struct bt_mesh_model *const vnd_models;
+};
+
+/* Foundation Models */
+#define BT_MESH_MODEL_ID_CFG_SRV 0x0000
+#define BT_MESH_MODEL_ID_CFG_CLI 0x0001
+#define BT_MESH_MODEL_ID_HEALTH_SRV 0x0002
+#define BT_MESH_MODEL_ID_HEALTH_CLI 0x0003
+
+/* Models from the Mesh Model Specification */
+#define BT_MESH_MODEL_ID_GEN_ONOFF_SRV 0x1000
+#define BT_MESH_MODEL_ID_GEN_ONOFF_CLI 0x1001
+#define BT_MESH_MODEL_ID_GEN_LEVEL_SRV 0x1002
+#define BT_MESH_MODEL_ID_GEN_LEVEL_CLI 0x1003
+#define BT_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_SRV 0x1004
+#define BT_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_CLI 0x1005
+#define BT_MESH_MODEL_ID_GEN_POWER_ONOFF_SRV 0x1006
+#define BT_MESH_MODEL_ID_GEN_POWER_ONOFF_SETUP_SRV 0x1007
+#define BT_MESH_MODEL_ID_GEN_POWER_ONOFF_CLI 0x1008
+#define BT_MESH_MODEL_ID_GEN_POWER_LEVEL_SRV 0x1009
+#define BT_MESH_MODEL_ID_GEN_POWER_LEVEL_SETUP_SRV 0x100a
+#define BT_MESH_MODEL_ID_GEN_POWER_LEVEL_CLI 0x100b
+#define BT_MESH_MODEL_ID_GEN_BATTERY_SRV 0x100c
+#define BT_MESH_MODEL_ID_GEN_BATTERY_CLI 0x100d
+#define BT_MESH_MODEL_ID_GEN_LOCATION_SRV 0x100e
+#define BT_MESH_MODEL_ID_GEN_LOCATION_SETUPSRV 0x100f
+#define BT_MESH_MODEL_ID_GEN_LOCATION_CLI 0x1010
+#define BT_MESH_MODEL_ID_GEN_ADMIN_PROP_SRV 0x1011
+#define BT_MESH_MODEL_ID_GEN_MANUFACTURER_PROP_SRV 0x1012
+#define BT_MESH_MODEL_ID_GEN_USER_PROP_SRV 0x1013
+#define BT_MESH_MODEL_ID_GEN_CLIENT_PROP_SRV 0x1014
+#define BT_MESH_MODEL_ID_GEN_PROP_CLI 0x1015
+#define BT_MESH_MODEL_ID_SENSOR_SRV 0x1100
+#define BT_MESH_MODEL_ID_SENSOR_SETUP_SRV 0x1101
+#define BT_MESH_MODEL_ID_SENSOR_CLI 0x1102
+#define BT_MESH_MODEL_ID_TIME_SRV 0x1200
+#define BT_MESH_MODEL_ID_TIME_SETUP_SRV 0x1201
+#define BT_MESH_MODEL_ID_TIME_CLI 0x1202
+#define BT_MESH_MODEL_ID_SCENE_SRV 0x1203
+#define BT_MESH_MODEL_ID_SCENE_SETUP_SRV 0x1204
+#define BT_MESH_MODEL_ID_SCENE_CLI 0x1205
+#define BT_MESH_MODEL_ID_SCHEDULER_SRV 0x1206
+#define BT_MESH_MODEL_ID_SCHEDULER_SETUP_SRV 0x1207
+#define BT_MESH_MODEL_ID_SCHEDULER_CLI 0x1208
+#define BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_SRV 0x1300
+#define BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_SETUP_SRV 0x1301
+#define BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_CLI 0x1302
+#define BT_MESH_MODEL_ID_LIGHT_CTL_SRV 0x1303
+#define BT_MESH_MODEL_ID_LIGHT_CTL_SETUP_SRV 0x1304
+#define BT_MESH_MODEL_ID_LIGHT_CTL_CLI 0x1305
+#define BT_MESH_MODEL_ID_LIGHT_CTL_TEMP_SRV 0x1306
+#define BT_MESH_MODEL_ID_LIGHT_HSL_SRV 0x1307
+#define BT_MESH_MODEL_ID_LIGHT_HSL_SETUP_SRV 0x1308
+#define BT_MESH_MODEL_ID_LIGHT_HSL_CLI 0x1309
+#define BT_MESH_MODEL_ID_LIGHT_HSL_HUE_SRV 0x130a
+#define BT_MESH_MODEL_ID_LIGHT_HSL_SAT_SRV 0x130b
+#define BT_MESH_MODEL_ID_LIGHT_XYL_SRV 0x130c
+#define BT_MESH_MODEL_ID_LIGHT_XYL_SETUP_SRV 0x130d
+#define BT_MESH_MODEL_ID_LIGHT_XYL_CLI 0x130e
+#define BT_MESH_MODEL_ID_LIGHT_LC_SRV 0x130f
+#define BT_MESH_MODEL_ID_LIGHT_LC_SETUPSRV 0x1310
+#define BT_MESH_MODEL_ID_LIGHT_LC_CLI 0x1311
+
+/** Message sending context. */
+struct bt_mesh_msg_ctx {
+ /** NetKey Index of the subnet to send the message on. */
+ u16_t net_idx;
+
+ /** AppKey Index to encrypt the message with. */
+ u16_t app_idx;
+
+ /** Remote address. */
+ u16_t addr;
+
+ /** Received TTL value. Not used for sending. */
+ u8_t recv_ttl: 7;
+
+ /** Force sending reliably by using segment acknowledgement */
+ u8_t send_rel: 1;
+
+ /** TTL, or BT_MESH_TTL_DEFAULT for default TTL. */
+ u8_t send_ttl;
+
+ /** Change by Espressif, opcode of a received message.
+ * Not used for sending message. */
+ u32_t recv_op;
+
+ /** Change by Espressif, destination address of a received
+ * message. Not used for sending message. */
+ u16_t recv_dst;
+
+ /** Change by Espressif, model corresponds to the message */
+ struct bt_mesh_model *model;
+
+ /** Change by Espressif, if the message is sent by a server
+ * model. Not used for receiving message. */
+ bool srv_send;
+};
+
+struct bt_mesh_model_op {
+ /* OpCode encoded using the BT_MESH_MODEL_OP_* macros */
+ const u32_t opcode;
+
+ /* Minimum required message length */
+ const size_t min_len;
+
+ /* Message handler for the opcode */
+ void (*const func)(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf);
+};
+
+#define BT_MESH_MODEL_OP_1(b0) (b0)
+#define BT_MESH_MODEL_OP_2(b0, b1) (((b0) << 8) | (b1))
+#define BT_MESH_MODEL_OP_3(b0, cid) ((((b0) << 16) | 0xc00000) | (cid))
+
+#define BT_MESH_MODEL_OP_END { 0, 0, NULL }
+#define BT_MESH_MODEL_NO_OPS ((struct bt_mesh_model_op []) \
+ { BT_MESH_MODEL_OP_END })
+
+/** Helper to define an empty model array */
+#define BT_MESH_MODEL_NONE ((struct bt_mesh_model []){})
+
+#define BT_MESH_MODEL(_id, _op, _pub, _user_data) \
+{ \
+ .id = (_id), \
+ .op = _op, \
+ .keys = { [0 ... (CONFIG_BT_MESH_MODEL_KEY_COUNT - 1)] = \
+ BT_MESH_KEY_UNUSED }, \
+ .pub = _pub, \
+ .groups = { [0 ... (CONFIG_BT_MESH_MODEL_GROUP_COUNT - 1)] = \
+ BT_MESH_ADDR_UNASSIGNED }, \
+ .user_data = _user_data, \
+}
+
+#define BT_MESH_MODEL_VND(_company, _id, _op, _pub, _user_data) \
+{ \
+ .vnd.company = (_company), \
+ .vnd.id = (_id), \
+ .op = _op, \
+ .pub = _pub, \
+ .keys = { [0 ... (CONFIG_BT_MESH_MODEL_KEY_COUNT - 1)] = \
+ BT_MESH_KEY_UNUSED }, \
+ .groups = { [0 ... (CONFIG_BT_MESH_MODEL_GROUP_COUNT - 1)] = \
+ BT_MESH_ADDR_UNASSIGNED }, \
+ .user_data = _user_data, \
+}
+
+/** @def BT_MESH_TRANSMIT
+ *
+ * @brief Encode transmission count & interval steps.
+ *
+ * @param count Number of retransmissions (first transmission is excluded).
+ * @param int_ms Interval steps in milliseconds. Must be greater than 0
+ * and a multiple of 10.
+ *
+ * @return Mesh transmit value that can be used e.g. for the default
+ * values of the configuration model data.
+ */
+#define BT_MESH_TRANSMIT(count, int_ms) ((count) | (((int_ms / 10) - 1) << 3))
+
+/** @def BT_MESH_TRANSMIT_COUNT
+ *
+ * @brief Decode transmit count from a transmit value.
+ *
+ * @param transmit Encoded transmit count & interval value.
+ *
+ * @return Transmission count (actual transmissions is N + 1).
+ */
+#define BT_MESH_TRANSMIT_COUNT(transmit) (((transmit) & (u8_t)BIT_MASK(3)))
+
+/** @def BT_MESH_TRANSMIT_INT
+ *
+ * @brief Decode transmit interval from a transmit value.
+ *
+ * @param transmit Encoded transmit count & interval value.
+ *
+ * @return Transmission interval in milliseconds.
+ */
+#define BT_MESH_TRANSMIT_INT(transmit) ((((transmit) >> 3) + 1) * 10)
+
+/** @def BT_MESH_PUB_TRANSMIT
+ *
+ * @brief Encode Publish Retransmit count & interval steps.
+ *
+ * @param count Number of retransmissions (first transmission is excluded).
+ * @param int_ms Interval steps in milliseconds. Must be greater than 0
+ * and a multiple of 50.
+ *
+ * @return Mesh transmit value that can be used e.g. for the default
+ * values of the configuration model data.
+ */
+#define BT_MESH_PUB_TRANSMIT(count, int_ms) BT_MESH_TRANSMIT(count, \
+ (int_ms) / 5)
+
+/** @def BT_MESH_PUB_TRANSMIT_COUNT
+ *
+ * @brief Decode Pubhlish Retransmit count from a given value.
+ *
+ * @param transmit Encoded Publish Retransmit count & interval value.
+ *
+ * @return Retransmission count (actual transmissions is N + 1).
+ */
+#define BT_MESH_PUB_TRANSMIT_COUNT(transmit) BT_MESH_TRANSMIT_COUNT(transmit)
+
+/** @def BT_MESH_PUB_TRANSMIT_INT
+ *
+ * @brief Decode Publish Retransmit interval from a given value.
+ *
+ * @param transmit Encoded Publish Retransmit count & interval value.
+ *
+ * @return Transmission interval in milliseconds.
+ */
+#define BT_MESH_PUB_TRANSMIT_INT(transmit) ((((transmit) >> 3) + 1) * 50)
+
+/** Model publication context. */
+struct bt_mesh_model_pub {
+ /** The model the context belongs to. Initialized by the stack. */
+ struct bt_mesh_model *mod;
+
+ u16_t addr; /**< Publish Address. */
+ u16_t key; /**< Publish AppKey Index. */
+
+ u8_t ttl; /**< Publish Time to Live. */
+ u8_t retransmit; /**< Retransmit Count & Interval Steps. */
+ u8_t period; /**< Publish Period. */
+ u8_t period_div: 4, /**< Divisor for the Period. */
+ cred: 1, /**< Friendship Credentials Flag. */
+ count: 3; /**< Retransmissions left. */
+
+ u32_t period_start; /**< Start of the current period. */
+
+ /** @brief Publication buffer, containing the publication message.
+ *
+ * The application is expected to initialize this with
+ * a valid net_buf_simple pointer, with the help of e.g.
+ * the NET_BUF_SIMPLE() macro. The publication buffer must
+ * contain a valid publication message before calling the
+ * bt_mesh_model_publish() API or after the publication's
+ * @ref bt_mesh_model_pub.update callback has been called
+ * and returned success. The buffer must be created outside
+ * of function context, i.e. it must not be on the stack.
+ * This is most conveniently acheived by creating it inline
+ * when declaring the publication context:
+ *
+ * static struct bt_mesh_model_pub my_pub = {
+ * .msg = NET_BUF_SIMPLE(size),
+ * };
+ */
+ struct net_buf_simple *msg;
+
+ /** @brief Callback for updating the publication buffer.
+ *
+ * When set to NULL, the model is assumed not to support
+ * periodic publishing. When set to non-NULL the callback
+ * will be called periodically and is expected to update
+ * @ref bt_mesh_model_pub.msg with a valid publication
+ * message.
+ *
+ * @param mod The Model the Publication Context belogs to.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+ int (*update)(struct bt_mesh_model *mod);
+
+ /* Change by Espressif, role of the device going to publish messages */
+ u8_t dev_role;
+
+ /** Publish Period Timer. Only for stack-internal use. */
+ struct k_delayed_work timer;
+};
+
+/** @def BT_MESH_MODEL_PUB_DEFINE
+ *
+ * Define a model publication context.
+ *
+ * @param _name Variable name given to the context.
+ * @param _update Optional message update callback (may be NULL).
+ * @param _msg_len Length of the publication message.
+ */
+#define BT_MESH_MODEL_PUB_DEFINE(_name, _update, _msg_len) \
+ NET_BUF_SIMPLE_DEFINE_STATIC(bt_mesh_pub_msg_##_name, _msg_len); \
+ static struct bt_mesh_model_pub _name = { \
+ .update = _update, \
+ .msg = &bt_mesh_pub_msg_##_name, \
+ }
+
+
+/** Abstraction that describes a Mesh Model instance */
+struct bt_mesh_model {
+ union {
+ const u16_t id;
+ struct {
+ u16_t company;
+ u16_t id;
+ } vnd;
+ };
+
+ /* The Element this Model belongs to */
+ struct bt_mesh_elem *elem;
+
+ /* Model Publication */
+ struct bt_mesh_model_pub *const pub;
+
+ /* AppKey List */
+ u16_t keys[CONFIG_BT_MESH_MODEL_KEY_COUNT];
+
+ /* Subscription List (group or virtual addresses) */
+ u16_t groups[CONFIG_BT_MESH_MODEL_GROUP_COUNT];
+
+ const struct bt_mesh_model_op *const op;
+
+ /* Model-specific user data */
+ void *user_data;
+
+ /* Internal information, mainly for persistent storage */
+ u8_t element_idx; /* Belongs to Nth element */
+ u8_t model_idx; /* Is the Nth model in the element */
+ u16_t flags; /* Information about what has changed */
+};
+
+struct bt_mesh_send_cb {
+ void (*start)(u16_t duration, int err, void *cb_data);
+ void (*end)(int err, void *cb_data);
+};
+
+void bt_mesh_model_msg_init(struct net_buf_simple *msg, u32_t opcode);
+
+/** Special TTL value to request using configured default TTL */
+#define BT_MESH_TTL_DEFAULT 0xff
+
+/** Maximum allowed TTL value */
+#define BT_MESH_TTL_MAX 0x7f
+
+/**
+ * @brief Send an Access Layer message.
+ *
+ * @param model Mesh (client) Model that the message belongs to.
+ * @param ctx Message context, includes keys, TTL, etc.
+ * @param msg Access Layer payload (the actual message to be sent).
+ * @param cb Optional "message sent" callback.
+ * @param cb_data User data to be passed to the callback.
+ *
+ * @return 0 on success, or (negative) error code on failure.
+ */
+int bt_mesh_model_send(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *msg,
+ const struct bt_mesh_send_cb *cb,
+ void *cb_data);
+
+/**
+ * @brief Send a model publication message.
+ *
+ * Before calling this function, the user needs to ensure that the model
+ * publication message (@ref bt_mesh_model_pub.msg) contains a valid
+ * message to be sent. Note that this API is only to be used for
+ * non-period publishing. For periodic publishing the app only needs
+ * to make sure that @ref bt_mesh_model_pub.msg contains a valid message
+ * whenever the @ref bt_mesh_model_pub.update callback is called.
+ *
+ * @param model Mesh (client) Model that's publishing the message.
+ *
+ * @return 0 on success, or (negative) error code on failure.
+ */
+int bt_mesh_model_publish(struct bt_mesh_model *model);
+
+/** Node Composition */
+struct bt_mesh_comp {
+ u16_t cid;
+ u16_t pid;
+ u16_t vid;
+
+ size_t elem_count;
+ struct bt_mesh_elem *elem;
+};
+
+/**
+ * @}
+ */
+
+#endif /* __BT_MESH_ACCESS_H */
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_aes_encrypt.h b/components/bt/ble_mesh/mesh_core/include/mesh_aes_encrypt.h
new file mode 100644
index 0000000000..70408c4389
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_aes_encrypt.h
@@ -0,0 +1,172 @@
+/* aes.h - TinyCrypt interface to an AES-128 implementation */
+
+/*
+ * Copyright (C) 2017 by Intel Corporation, All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * - Redistributions of source code must retain the above copyright notice,
+ * this list of conditions and the following disclaimer.
+ *
+ * - Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * - Neither the name of Intel Corporation nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
+/**
+ * @file
+ * @brief -- Interface to an AES-128 implementation.
+ *
+ * Overview: AES-128 is a NIST approved block cipher specified in
+ * FIPS 197. Block ciphers are deterministic algorithms that
+ * perform a transformation specified by a symmetric key in fixed-
+ * length data sets, also called blocks.
+ *
+ * Security: AES-128 provides approximately 128 bits of security.
+ *
+ * Usage: 1) call tc_aes128_set_encrypt/decrypt_key to set the key.
+ *
+ * 2) call tc_aes_encrypt/decrypt to process the data.
+ */
+
+#ifndef __TC_AES_H__
+#define __TC_AES_H__
+
+#include
+#include
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#define Nb (4) /* number of columns (32-bit words) comprising the state */
+#define Nk (4) /* number of 32-bit words comprising the key */
+#define Nr (10) /* number of rounds */
+#define TC_AES_BLOCK_SIZE (Nb*Nk)
+#define TC_AES_KEY_SIZE (Nb*Nk)
+
+#define TC_CRYPTO_SUCCESS 1
+#define TC_CRYPTO_FAIL 0
+
+#define TC_ZERO_BYTE 0x00
+
+/* padding for last message block */
+#define TC_CMAC_PADDING 0x80
+
+typedef struct tc_aes_key_sched_struct {
+ unsigned int words[Nb * (Nr + 1)];
+} *TCAesKeySched_t;
+
+/* struct tc_cmac_struct represents the state of a CMAC computation */
+typedef struct tc_cmac_struct {
+ /* initialization vector */
+ uint8_t iv[TC_AES_BLOCK_SIZE];
+ /* used if message length is a multiple of block_size bytes */
+ uint8_t K1[TC_AES_BLOCK_SIZE];
+ /* used if message length isn't a multiple block_size bytes */
+ uint8_t K2[TC_AES_BLOCK_SIZE];
+ /* where to put bytes that didn't fill a block */
+ uint8_t leftover[TC_AES_BLOCK_SIZE];
+ /* identifies the encryption key */
+ unsigned int keyid;
+ /* next available leftover location */
+ unsigned int leftover_offset;
+ /* AES key schedule */
+ TCAesKeySched_t sched;
+ /* calls to tc_cmac_update left before re-key */
+ uint64_t countdown;
+} *TCCmacState_t;
+
+
+/**
+ * @brief Set AES-128 encryption key
+ * Uses key k to initialize s
+ * @return returns TC_CRYPTO_SUCCESS (1)
+ * returns TC_CRYPTO_FAIL (0) if: s == NULL or k == NULL
+ * @note This implementation skips the additional steps required for keys
+ * larger than 128 bits, and must not be used for AES-192 or
+ * AES-256 key schedule -- see FIPS 197 for details
+ * @param s IN/OUT -- initialized struct tc_aes_key_sched_struct
+ * @param k IN -- points to the AES key
+ */
+int tc_aes128_set_encrypt_key(TCAesKeySched_t s, const uint8_t *k);
+
+/**
+ * @brief AES-128 Encryption procedure
+ * Encrypts contents of in buffer into out buffer under key;
+ * schedule s
+ * @note Assumes s was initialized by aes_set_encrypt_key;
+ * out and in point to 16 byte buffers
+ * @return returns TC_CRYPTO_SUCCESS (1)
+ * returns TC_CRYPTO_FAIL (0) if: out == NULL or in == NULL or s == NULL
+ * @param out IN/OUT -- buffer to receive ciphertext block
+ * @param in IN -- a plaintext block to encrypt
+ * @param s IN -- initialized AES key schedule
+ */
+int tc_aes_encrypt(uint8_t *out, const uint8_t *in,
+ const TCAesKeySched_t s);
+
+/**
+ * @brief Set the AES-128 decryption key
+ * Uses key k to initialize s
+ * @return returns TC_CRYPTO_SUCCESS (1)
+ * returns TC_CRYPTO_FAIL (0) if: s == NULL or k == NULL
+ * @note This is the implementation of the straightforward inverse cipher
+ * using the cipher documented in FIPS-197 figure 12, not the
+ * equivalent inverse cipher presented in Figure 15
+ * @warning This routine skips the additional steps required for keys larger
+ * than 128, and must not be used for AES-192 or AES-256 key
+ * schedule -- see FIPS 197 for details
+ * @param s IN/OUT -- initialized struct tc_aes_key_sched_struct
+ * @param k IN -- points to the AES key
+ */
+int tc_aes128_set_decrypt_key(TCAesKeySched_t s, const uint8_t *k);
+
+/**
+ * @brief AES-128 Encryption procedure
+ * Decrypts in buffer into out buffer under key schedule s
+ * @return returns TC_CRYPTO_SUCCESS (1)
+ * returns TC_CRYPTO_FAIL (0) if: out is NULL or in is NULL or s is NULL
+ * @note Assumes s was initialized by aes_set_encrypt_key
+ * out and in point to 16 byte buffers
+ * @param out IN/OUT -- buffer to receive ciphertext block
+ * @param in IN -- a plaintext block to encrypt
+ * @param s IN -- initialized AES key schedule
+ */
+int tc_aes_decrypt(uint8_t *out, const uint8_t *in,
+ const TCAesKeySched_t s);
+
+int tc_cmac_setup(TCCmacState_t s, const uint8_t *key, TCAesKeySched_t sched);
+
+void gf_double(uint8_t *out, uint8_t *in);
+
+int tc_cmac_init(TCCmacState_t s);
+
+int tc_cmac_update(TCCmacState_t s, const uint8_t *data, size_t data_length);
+
+int tc_cmac_final(uint8_t *tag, TCCmacState_t s);
+
+int tc_cmac_erase(TCCmacState_t s);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __TC_AES_H__ */
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_atomic.h b/components/bt/ble_mesh/mesh_core/include/mesh_atomic.h
new file mode 100644
index 0000000000..fcb7041397
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_atomic.h
@@ -0,0 +1,420 @@
+/* atomic operations */
+
+/*
+ * Copyright (c) 1997-2015, Wind River Systems, Inc.
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __ATOMIC_H__
+#define __ATOMIC_H__
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef int atomic_t;
+typedef atomic_t atomic_val_t;
+
+/**
+ * @defgroup atomic_apis Atomic Services APIs
+ * @ingroup kernel_apis
+ * @{
+ */
+
+/**
+ * @brief Atomic compare-and-set.
+ *
+ * This routine performs an atomic compare-and-set on @a target. If the current
+ * value of @a target equals @a old_value, @a target is set to @a new_value.
+ * If the current value of @a target does not equal @a old_value, @a target
+ * is left unchanged.
+ *
+ * @param target Address of atomic variable.
+ * @param old_value Original value to compare against.
+ * @param new_value New value to store.
+ * @return 1 if @a new_value is written, 0 otherwise.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline int atomic_cas(atomic_t *target, atomic_val_t old_value,
+ atomic_val_t new_value)
+{
+ return __atomic_compare_exchange_n(target, &old_value, new_value,
+ 0, __ATOMIC_SEQ_CST,
+ __ATOMIC_SEQ_CST);
+}
+#else
+extern int atomic_cas(atomic_t *target, atomic_val_t old_value,
+ atomic_val_t new_value);
+#endif
+
+/**
+ *
+ * @brief Atomic addition.
+ *
+ * This routine performs an atomic addition on @a target.
+ *
+ * @param target Address of atomic variable.
+ * @param value Value to add.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_add(atomic_t *target, atomic_val_t value)
+{
+ return __atomic_fetch_add(target, value, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_add(atomic_t *target, atomic_val_t value);
+#endif
+
+/**
+ *
+ * @brief Atomic subtraction.
+ *
+ * This routine performs an atomic subtraction on @a target.
+ *
+ * @param target Address of atomic variable.
+ * @param value Value to subtract.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_sub(atomic_t *target, atomic_val_t value)
+{
+ return __atomic_fetch_sub(target, value, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_sub(atomic_t *target, atomic_val_t value);
+#endif
+
+/**
+ *
+ * @brief Atomic increment.
+ *
+ * This routine performs an atomic increment by 1 on @a target.
+ *
+ * @param target Address of atomic variable.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_inc(atomic_t *target)
+{
+ return atomic_add(target, 1);
+}
+#else
+extern atomic_val_t atomic_inc(atomic_t *target);
+#endif
+
+/**
+ *
+ * @brief Atomic decrement.
+ *
+ * This routine performs an atomic decrement by 1 on @a target.
+ *
+ * @param target Address of atomic variable.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_dec(atomic_t *target)
+{
+ return atomic_sub(target, 1);
+}
+#else
+extern atomic_val_t atomic_dec(atomic_t *target);
+#endif
+
+/**
+ *
+ * @brief Atomic get.
+ *
+ * This routine performs an atomic read on @a target.
+ *
+ * @param target Address of atomic variable.
+ *
+ * @return Value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_get(const atomic_t *target)
+{
+ return __atomic_load_n(target, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_get(const atomic_t *target);
+#endif
+
+/**
+ *
+ * @brief Atomic get-and-set.
+ *
+ * This routine atomically sets @a target to @a value and returns
+ * the previous value of @a target.
+ *
+ * @param target Address of atomic variable.
+ * @param value Value to write to @a target.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_set(atomic_t *target, atomic_val_t value)
+{
+ /* This builtin, as described by Intel, is not a traditional
+ * test-and-set operation, but rather an atomic exchange operation. It
+ * writes value into *ptr, and returns the previous contents of *ptr.
+ */
+ return __atomic_exchange_n(target, value, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_set(atomic_t *target, atomic_val_t value);
+#endif
+
+/**
+ *
+ * @brief Atomic clear.
+ *
+ * This routine atomically sets @a target to zero and returns its previous
+ * value. (Hence, it is equivalent to atomic_set(target, 0).)
+ *
+ * @param target Address of atomic variable.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_clear(atomic_t *target)
+{
+ return atomic_set(target, 0);
+}
+#else
+extern atomic_val_t atomic_clear(atomic_t *target);
+#endif
+
+/**
+ *
+ * @brief Atomic bitwise inclusive OR.
+ *
+ * This routine atomically sets @a target to the bitwise inclusive OR of
+ * @a target and @a value.
+ *
+ * @param target Address of atomic variable.
+ * @param value Value to OR.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_or(atomic_t *target, atomic_val_t value)
+{
+ return __atomic_fetch_or(target, value, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_or(atomic_t *target, atomic_val_t value);
+#endif
+
+/**
+ *
+ * @brief Atomic bitwise exclusive OR (XOR).
+ *
+ * This routine atomically sets @a target to the bitwise exclusive OR (XOR) of
+ * @a target and @a value.
+ *
+ * @param target Address of atomic variable.
+ * @param value Value to XOR
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_xor(atomic_t *target, atomic_val_t value)
+{
+ return __atomic_fetch_xor(target, value, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_xor(atomic_t *target, atomic_val_t value);
+#endif
+
+/**
+ *
+ * @brief Atomic bitwise AND.
+ *
+ * This routine atomically sets @a target to the bitwise AND of @a target
+ * and @a value.
+ *
+ * @param target Address of atomic variable.
+ * @param value Value to AND.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_and(atomic_t *target, atomic_val_t value)
+{
+ return __atomic_fetch_and(target, value, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_and(atomic_t *target, atomic_val_t value);
+#endif
+
+/**
+ *
+ * @brief Atomic bitwise NAND.
+ *
+ * This routine atomically sets @a target to the bitwise NAND of @a target
+ * and @a value. (This operation is equivalent to target = ~(target & value).)
+ *
+ * @param target Address of atomic variable.
+ * @param value Value to NAND.
+ *
+ * @return Previous value of @a target.
+ */
+#ifdef CONFIG_ATOMIC_OPERATIONS_BUILTIN
+static inline atomic_val_t atomic_nand(atomic_t *target, atomic_val_t value)
+{
+ return __atomic_fetch_nand(target, value, __ATOMIC_SEQ_CST);
+}
+#else
+extern atomic_val_t atomic_nand(atomic_t *target, atomic_val_t value);
+#endif
+
+
+/**
+ * @brief Initialize an atomic variable.
+ *
+ * This macro can be used to initialize an atomic variable. For example,
+ * @code atomic_t my_var = ATOMIC_INIT(75); @endcode
+ *
+ * @param i Value to assign to atomic variable.
+ */
+#define ATOMIC_INIT(i) (i)
+
+/**
+ * @cond INTERNAL_HIDDEN
+ */
+
+#define ATOMIC_BITS (sizeof(atomic_val_t) * 8)
+#define ATOMIC_MASK(bit) (1 << ((bit) & (ATOMIC_BITS - 1)))
+#define ATOMIC_ELEM(addr, bit) ((addr) + ((bit) / ATOMIC_BITS))
+
+/**
+ * INTERNAL_HIDDEN @endcond
+ */
+
+/**
+ * @brief Define an array of atomic variables.
+ *
+ * This macro defines an array of atomic variables containing at least
+ * @a num_bits bits.
+ *
+ * @note
+ * If used from file scope, the bits of the array are initialized to zero;
+ * if used from within a function, the bits are left uninitialized.
+ *
+ * @param name Name of array of atomic variables.
+ * @param num_bits Number of bits needed.
+ */
+#define ATOMIC_DEFINE(name, num_bits) \
+ atomic_t name[1 + ((num_bits) - 1) / ATOMIC_BITS]
+
+/**
+ * @brief Atomically test a bit.
+ *
+ * This routine tests whether bit number @a bit of @a target is set or not.
+ * The target may be a single atomic variable or an array of them.
+ *
+ * @param target Address of atomic variable or array.
+ * @param bit Bit number (starting from 0).
+ *
+ * @return 1 if the bit was set, 0 if it wasn't.
+ */
+static inline int atomic_test_bit(const atomic_t *target, int bit)
+{
+ atomic_val_t val = atomic_get(ATOMIC_ELEM(target, bit));
+
+ return (1 & (val >> (bit & (ATOMIC_BITS - 1))));
+}
+
+/**
+ * @brief Atomically test and clear a bit.
+ *
+ * Atomically clear bit number @a bit of @a target and return its old value.
+ * The target may be a single atomic variable or an array of them.
+ *
+ * @param target Address of atomic variable or array.
+ * @param bit Bit number (starting from 0).
+ *
+ * @return 1 if the bit was set, 0 if it wasn't.
+ */
+static inline int atomic_test_and_clear_bit(atomic_t *target, int bit)
+{
+ atomic_val_t mask = ATOMIC_MASK(bit);
+ atomic_val_t old;
+
+ old = atomic_and(ATOMIC_ELEM(target, bit), ~mask);
+
+ return (old & mask) != 0;
+}
+
+/**
+ * @brief Atomically set a bit.
+ *
+ * Atomically set bit number @a bit of @a target and return its old value.
+ * The target may be a single atomic variable or an array of them.
+ *
+ * @param target Address of atomic variable or array.
+ * @param bit Bit number (starting from 0).
+ *
+ * @return 1 if the bit was set, 0 if it wasn't.
+ */
+static inline int atomic_test_and_set_bit(atomic_t *target, int bit)
+{
+ atomic_val_t mask = ATOMIC_MASK(bit);
+ atomic_val_t old;
+
+ old = atomic_or(ATOMIC_ELEM(target, bit), mask);
+
+ return (old & mask) != 0;
+}
+
+/**
+ * @brief Atomically clear a bit.
+ *
+ * Atomically clear bit number @a bit of @a target.
+ * The target may be a single atomic variable or an array of them.
+ *
+ * @param target Address of atomic variable or array.
+ * @param bit Bit number (starting from 0).
+ *
+ * @return N/A
+ */
+static inline void atomic_clear_bit(atomic_t *target, int bit)
+{
+ atomic_val_t mask = ATOMIC_MASK(bit);
+
+ atomic_and(ATOMIC_ELEM(target, bit), ~mask);
+}
+
+/**
+ * @brief Atomically set a bit.
+ *
+ * Atomically set bit number @a bit of @a target.
+ * The target may be a single atomic variable or an array of them.
+ *
+ * @param target Address of atomic variable or array.
+ * @param bit Bit number (starting from 0).
+ *
+ * @return N/A
+ */
+static inline void atomic_set_bit(atomic_t *target, int bit)
+{
+ atomic_val_t mask = ATOMIC_MASK(bit);
+
+ atomic_or(ATOMIC_ELEM(target, bit), mask);
+}
+
+/**
+ * @}
+ */
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __ATOMIC_H__ */
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_bearer_adapt.h b/components/bt/ble_mesh/mesh_core/include/mesh_bearer_adapt.h
new file mode 100644
index 0000000000..db158cae3f
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_bearer_adapt.h
@@ -0,0 +1,836 @@
+/*
+ * Copyright (c) 2017 Nordic Semiconductor ASA
+ * Copyright (c) 2015-2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef _MESH_BEARER_ADRPT_H_
+#define _MESH_BEARER_ADRPT_H_
+
+#include
+#include "mesh_types.h"
+#include "mesh_buf.h"
+#include "mesh_bt_uuid.h"
+
+/* Advertising types */
+#define BT_LE_ADV_IND 0x00
+#define BT_LE_ADV_DIRECT_IND 0x01
+#define BT_LE_ADV_SCAN_IND 0x02
+#define BT_LE_ADV_NONCONN_IND 0x03
+#define BT_LE_ADV_DIRECT_IND_LOW_DUTY 0x04
+/* Needed in advertising reports when getting info about */
+#define BT_LE_ADV_SCAN_RSP 0x04
+
+#define BT_HCI_LE_SCAN_PASSIVE 0x00
+#define BT_HCI_LE_SCAN_ACTIVE 0x01
+
+#define BT_HCI_LE_SCAN_DISABLE 0x00
+#define BT_HCI_LE_SCAN_ENABLE 0x01
+
+#define BT_HCI_LE_SCAN_FILTER_DUP_DISABLE 0x00
+#define BT_HCI_LE_SCAN_FILTER_DUP_ENABLE 0x01
+
+/* Error codes for Error response PDU */
+#define BT_ATT_ERR_INVALID_HANDLE 0x01
+#define BT_ATT_ERR_READ_NOT_PERMITTED 0x02
+#define BT_ATT_ERR_WRITE_NOT_PERMITTED 0x03
+#define BT_ATT_ERR_INVALID_PDU 0x04
+#define BT_ATT_ERR_AUTHENTICATION 0x05
+#define BT_ATT_ERR_NOT_SUPPORTED 0x06
+#define BT_ATT_ERR_INVALID_OFFSET 0x07
+#define BT_ATT_ERR_AUTHORIZATION 0x08
+#define BT_ATT_ERR_PREPARE_QUEUE_FULL 0x09
+#define BT_ATT_ERR_ATTRIBUTE_NOT_FOUND 0x0a
+#define BT_ATT_ERR_ATTRIBUTE_NOT_LONG 0x0b
+#define BT_ATT_ERR_ENCRYPTION_KEY_SIZE 0x0c
+#define BT_ATT_ERR_INVALID_ATTRIBUTE_LEN 0x0d
+#define BT_ATT_ERR_UNLIKELY 0x0e
+#define BT_ATT_ERR_INSUFFICIENT_ENCRYPTION 0x0f
+#define BT_ATT_ERR_UNSUPPORTED_GROUP_TYPE 0x10
+#define BT_ATT_ERR_INSUFFICIENT_RESOURCES 0x11
+
+/* Common Profile Error Codes (from CSS) */
+#define BT_ATT_ERR_WRITE_REQ_REJECTED 0xfc
+#define BT_ATT_ERR_CCC_IMPROPER_CONF 0xfd
+#define BT_ATT_ERR_PROCEDURE_IN_PROGRESS 0xfe
+#define BT_ATT_ERR_OUT_OF_RANGE 0xff
+
+/* EIR/AD data type definitions */
+#define BT_DATA_FLAGS 0x01 /* AD flags */
+#define BT_DATA_UUID16_SOME 0x02 /* 16-bit UUID, more available */
+#define BT_DATA_UUID16_ALL 0x03 /* 16-bit UUID, all listed */
+#define BT_DATA_UUID32_SOME 0x04 /* 32-bit UUID, more available */
+#define BT_DATA_UUID32_ALL 0x05 /* 32-bit UUID, all listed */
+#define BT_DATA_UUID128_SOME 0x06 /* 128-bit UUID, more available */
+#define BT_DATA_UUID128_ALL 0x07 /* 128-bit UUID, all listed */
+#define BT_DATA_NAME_SHORTENED 0x08 /* Shortened name */
+#define BT_DATA_NAME_COMPLETE 0x09 /* Complete name */
+#define BT_DATA_TX_POWER 0x0a /* Tx Power */
+#define BT_DATA_SOLICIT16 0x14 /* Solicit UUIDs, 16-bit */
+#define BT_DATA_SOLICIT128 0x15 /* Solicit UUIDs, 128-bit */
+#define BT_DATA_SVC_DATA16 0x16 /* Service data, 16-bit UUID */
+#define BT_DATA_GAP_APPEARANCE 0x19 /* GAP appearance */
+#define BT_DATA_SOLICIT32 0x1f /* Solicit UUIDs, 32-bit */
+#define BT_DATA_SVC_DATA32 0x20 /* Service data, 32-bit UUID */
+#define BT_DATA_SVC_DATA128 0x21 /* Service data, 128-bit UUID */
+#define BT_DATA_MESH_PROV 0x29 /* Mesh Provisioning PDU */
+#define BT_DATA_MESH_MESSAGE 0x2a /* Mesh Networking PDU */
+#define BT_DATA_MESH_BEACON 0x2b /* Mesh Beacon */
+
+#define BT_DATA_MANUFACTURER_DATA 0xff /* Manufacturer Specific Data */
+
+#define BT_LE_AD_LIMITED 0x01 /* Limited Discoverable */
+#define BT_LE_AD_GENERAL 0x02 /* General Discoverable */
+#define BT_LE_AD_NO_BREDR 0x04 /* BR/EDR not supported */
+
+/* Client Characteristic Configuration Values */
+
+/** @def BT_GATT_CCC_NOTIFY
+ * @brief Client Characteristic Configuration Notification.
+ *
+ * If set, changes to Characteristic Value shall be notified.
+ */
+#define BT_GATT_CCC_NOTIFY 0x0001
+/** @def BT_GATT_CCC_INDICATE
+ * @brief Client Characteristic Configuration Indication.
+ *
+ * If set, changes to Characteristic Value shall be indicated.
+ */
+#define BT_GATT_CCC_INDICATE 0x0002
+
+
+/** @def BT_GATT_ERR
+ * @brief Construct error return value for attribute read and write callbacks.
+ *
+ * @param _att_err ATT error code
+ *
+ * @return Appropriate error code for the attribute callbacks.
+ *
+ */
+#define BT_GATT_ERR(_att_err) (-(_att_err))
+
+enum {
+ BT_GATT_ITER_STOP = 0,
+ BT_GATT_ITER_CONTINUE,
+};
+
+/* GATT attribute permission bit field values */
+enum {
+ /** No operations supported, e.g. for notify-only */
+ BT_GATT_PERM_NONE = 0,
+
+ /** Attribute read permission. */
+ BT_GATT_PERM_READ = BIT(0),
+
+ /** Attribute write permission. */
+ BT_GATT_PERM_WRITE = BIT(1),
+
+ /** Attribute read permission with encryption.
+ *
+ * If set, requires encryption for read access.
+ */
+ BT_GATT_PERM_READ_ENCRYPT = BIT(2),
+
+ /** Attribute write permission with encryption.
+ *
+ * If set, requires encryption for write access.
+ */
+ BT_GATT_PERM_WRITE_ENCRYPT = BIT(3),
+
+ /** Attribute read permission with authentication.
+ *
+ * If set, requires encryption using authenticated link-key for read
+ * access.
+ */
+ BT_GATT_PERM_READ_AUTHEN = BIT(4),
+
+ /** Attribute write permission with authentication.
+ *
+ * If set, requires encryption using authenticated link-key for write
+ * access.
+ */
+ BT_GATT_PERM_WRITE_AUTHEN = BIT(5),
+
+ /** Attribute prepare write permission.
+ *
+ * If set, allows prepare writes with use of BT_GATT_WRITE_FLAG_PREPARE
+ * passed to write callback.
+ */
+ BT_GATT_PERM_PREPARE_WRITE = BIT(6),
+};
+
+/** Advertising options */
+enum {
+ /** Convenience value when no options are specified. */
+ BT_LE_ADV_OPT_NONE = 0,
+
+ /** Advertise as connectable. Type of advertising is determined by
+ * providing SCAN_RSP data and/or enabling local privacy support.
+ */
+ BT_LE_ADV_OPT_CONNECTABLE = BIT(0),
+
+ /** Don't try to resume connectable advertising after a connection.
+ * This option is only meaningful when used together with
+ * BT_LE_ADV_OPT_CONNECTABLE. If set the advertising will be stopped
+ * when bt_le_adv_stop() is called or when an incoming (slave)
+ * connection happens. If this option is not set the stack will
+ * take care of keeping advertising enabled even as connections
+ * occur.
+ */
+ BT_LE_ADV_OPT_ONE_TIME = BIT(1),
+};
+
+/* Defined GAP timers */
+#define BT_GAP_SCAN_FAST_INTERVAL 0x0060 /* 60 ms */
+#define BT_GAP_SCAN_FAST_WINDOW 0x0030 /* 30 ms */
+#define BT_GAP_SCAN_SLOW_INTERVAL_1 0x0800 /* 1.28 s */
+#define BT_GAP_SCAN_SLOW_WINDOW_1 0x0012 /* 11.25 ms */
+#define BT_GAP_SCAN_SLOW_INTERVAL_2 0x1000 /* 2.56 s */
+#define BT_GAP_SCAN_SLOW_WINDOW_2 0x0012 /* 11.25 ms */
+#define BT_GAP_ADV_FAST_INT_MIN_0 0x0020 /* 20 ms */
+#define BT_GAP_ADV_FAST_INT_MAX_0 0x0020 /* 20 ms */
+#define BT_GAP_ADV_FAST_INT_MIN_1 0x0030 /* 30 ms */
+#define BT_GAP_ADV_FAST_INT_MAX_1 0x0060 /* 60 ms */
+#define BT_GAP_ADV_FAST_INT_MIN_2 0x00a0 /* 100 ms */
+#define BT_GAP_ADV_FAST_INT_MAX_2 0x00f0 /* 150 ms */
+#define BT_GAP_ADV_SLOW_INT_MIN 0x0320 //0x0640 /* 1 s */
+#define BT_GAP_ADV_SLOW_INT_MAX 0x0320 //0x0780 /* 1.2 s */
+#define BT_GAP_INIT_CONN_INT_MIN 0x0018 /* 30 ms */
+#define BT_GAP_INIT_CONN_INT_MAX 0x0028 /* 50 ms */
+
+
+
+/* Characteristic Properties Bit field values */
+
+/** @def BT_GATT_CHRC_BROADCAST
+ * @brief Characteristic broadcast property.
+ *
+ * If set, permits broadcasts of the Characteristic Value using Server
+ * Characteristic Configuration Descriptor.
+ */
+#define BT_GATT_CHRC_BROADCAST 0x01
+/** @def BT_GATT_CHRC_READ
+ * @brief Characteristic read property.
+ *
+ * If set, permits reads of the Characteristic Value.
+ */
+#define BT_GATT_CHRC_READ 0x02
+/** @def BT_GATT_CHRC_WRITE_WITHOUT_RESP
+ * @brief Characteristic write without response property.
+ *
+ * If set, permits write of the Characteristic Value without response.
+ */
+#define BT_GATT_CHRC_WRITE_WITHOUT_RESP 0x04
+/** @def BT_GATT_CHRC_WRITE
+ * @brief Characteristic write with response property.
+ *
+ * If set, permits write of the Characteristic Value with response.
+ */
+#define BT_GATT_CHRC_WRITE 0x08
+/** @def BT_GATT_CHRC_NOTIFY
+ * @brief Characteristic notify property.
+ *
+ * If set, permits notifications of a Characteristic Value without
+ * acknowledgment.
+ */
+#define BT_GATT_CHRC_NOTIFY 0x10
+/** @def BT_GATT_CHRC_INDICATE
+ * @brief Characteristic indicate property.
+ *
+ * If set, permits indications of a Characteristic Value with acknowledgment.
+ */
+#define BT_GATT_CHRC_INDICATE 0x20
+/** @def BT_GATT_CHRC_AUTH
+ * @brief Characteristic Authenticated Signed Writes property.
+ *
+ * If set, permits signed writes to the Characteristic Value.
+ */
+#define BT_GATT_CHRC_AUTH 0x40
+/** @def BT_GATT_CHRC_EXT_PROP
+ * @brief Characteristic Extended Properties property.
+ *
+ * If set, additional characteristic properties are defined in the
+ * Characteristic Extended Properties Descriptor.
+ */
+#define BT_GATT_CHRC_EXT_PROP 0x80
+
+
+/** @brief Characteristic Attribute Value. */
+struct bt_gatt_chrc {
+ /** Characteristic UUID. */
+ const struct bt_uuid *uuid;
+ /** Characteristic properties. */
+ u8_t properties;
+};
+
+/** @brief GATT Service structure */
+struct bt_gatt_service {
+ /** Service Attributes */
+ struct bt_gatt_attr *attrs;
+ /** Service Attribute count */
+ u16_t attr_count;
+ sys_snode_t node;
+};
+
+/*
+ * Copyright (c) 2015 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+typedef enum __packed {
+ BT_CONN_DISCONNECTED,
+ BT_CONN_CONNECT_SCAN,
+ BT_CONN_CONNECT,
+ BT_CONN_CONNECTED,
+ BT_CONN_DISCONNECT,
+} bt_conn_state_t;
+
+/** Security level. */
+typedef enum __packed {
+ /** Only for BR/EDR special cases, like SDP */
+ BT_SECURITY_NONE,
+ /** No encryption and no authentication. */
+ BT_SECURITY_LOW,
+ /** Encryption and no authentication (no MITM). */
+ BT_SECURITY_MEDIUM,
+ /** Encryption and authentication (MITM). */
+ BT_SECURITY_HIGH,
+ /** Authenticated Secure Connections */
+ BT_SECURITY_FIPS,
+} bt_security_t;
+
+struct ecb_param {
+ u8_t key[16];
+ u8_t clear_text[16];
+ u8_t cipher_text[16];
+} __packed;
+
+typedef struct {
+ u8_t val[6];
+} bt_addr_t;
+
+typedef struct {
+ u8_t type;
+ bt_addr_t a;
+} bt_addr_le_t;
+
+/** Description of different data types that can be encoded into
+ * advertising data. Used to form arrays that are passed to the
+ * bt_le_adv_start() function.
+ */
+struct bt_data {
+ u8_t type;
+ u8_t data_len;
+ const u8_t *data;
+};
+
+/** @brief Helper to declare elements of bt_data arrays
+ *
+ * This macro is mainly for creating an array of struct bt_data
+ * elements which is then passed to bt_le_adv_start().
+ *
+ * @param _type Type of advertising data field
+ * @param _data Pointer to the data field payload
+ * @param _data_len Number of bytes behind the _data pointer
+ */
+#define BT_DATA(_type, _data, _data_len) \
+ { \
+ .type = (_type), \
+ .data_len = (_data_len), \
+ .data = (const u8_t *)(_data), \
+ }
+
+/** @brief Helper to declare elements of bt_data arrays
+ *
+ * This macro is mainly for creating an array of struct bt_data
+ * elements which is then passed to bt_le_adv_start().
+ *
+ * @param _type Type of advertising data field
+ * @param _bytes Variable number of single-byte parameters
+ */
+#define BT_DATA_BYTES(_type, _bytes...) \
+ BT_DATA(_type, ((u8_t []) { _bytes }), \
+ sizeof((u8_t []) { _bytes }))
+
+/** LE Advertising Parameters. */
+struct bt_le_adv_param {
+ /** Bit-field of advertising options */
+ u8_t options;
+
+ /** Minimum Advertising Interval (N * 0.625) */
+ u16_t interval_min;
+
+ /** Maximum Advertising Interval (N * 0.625) */
+ u16_t interval_max;
+
+ /** Optional predefined (random) own address. Currently
+ * the only permitted use of this is for NRPA with
+ * non-connectable advertising.
+ */
+ const bt_addr_t *own_addr;
+};
+
+/** LE scan parameters */
+struct bt_le_scan_param {
+ /** Scan type (BT_HCI_LE_SCAN_ACTIVE or BT_HCI_LE_SCAN_PASSIVE) */
+ u8_t type;
+
+ /** Duplicate filtering (BT_HCI_LE_SCAN_FILTER_DUP_ENABLE or
+ * BT_HCI_LE_SCAN_FILTER_DUP_DISABLE)
+ */
+ u8_t filter_dup;
+
+ /** Scan interval (N * 0.625 ms) */
+ u16_t interval;
+
+ /** Scan window (N * 0.625 ms) */
+ u16_t window;
+};
+
+/** Connection parameters for LE connections */
+struct bt_le_conn_param {
+ u16_t interval_min;
+ u16_t interval_max;
+ u16_t latency;
+ u16_t timeout;
+};
+
+struct bt_conn {
+ u16_t handle;
+ u8_t type;
+ u8_t role;
+
+ /* Connection error or reason for disconnect */
+ u8_t err;
+
+ bt_conn_state_t state;
+
+ u16_t rx_len;
+ struct net_buf *rx;
+
+ /* Sent but not acknowledged TX packets */
+ sys_slist_t tx_pending;
+ /* Acknowledged but not yet notified TX packets */
+ struct k_fifo tx_notify;
+
+ /* Queue for outgoing ACL data */
+ struct k_fifo tx_queue;
+
+ /* Active L2CAP channels */
+ sys_slist_t channels;
+
+ atomic_t ref;
+};
+
+
+/** @typedef bt_le_scan_cb_t
+ * @brief Callback type for reporting LE scan results.
+ *
+ * A function of this type is given to the bt_le_scan_start() function
+ * and will be called for any discovered LE device.
+ *
+ * @param addr Advertiser LE address and type.
+ * @param rssi Strength of advertiser signal.
+ * @param adv_type Type of advertising response from advertiser.
+ * @param data Buffer containing advertiser data.
+ */
+typedef void bt_le_scan_cb_t(const bt_addr_le_t *addr, s8_t rssi,
+ u8_t adv_type, struct net_buf_simple *buf);
+
+/* @typedef bt_dh_key_cb_t
+ * @brief Callback type for DH Key calculation.
+ *
+ * Used to notify of the calculated DH Key.
+ *
+ * @param key The DH Key, or NULL in case of failure.
+ */
+typedef void (*bt_dh_key_cb_t)(const u8_t key[32]);
+
+/** @typedef bt_gatt_attr_func_t
+ * @brief Attribute iterator callback.
+ *
+ * @param attr Attribute found.
+ * @param user_data Data given.
+ *
+ * @return BT_GATT_ITER_CONTINUE if should continue to the next attribute
+ * or BT_GATT_ITER_STOP to stop.
+ */
+typedef u8_t (*bt_gatt_attr_func_t)(const struct bt_gatt_attr *attr,
+ void *user_data);
+
+/** @brief Connection callback structure.
+ *
+ * This structure is used for tracking the state of a connection.
+ * It is registered with the help of the bt_mesh_gatts_conn_cb_register() API.
+ * It's permissible to register multiple instances of this @ref bt_conn_cb
+ * type, in case different modules of an application are interested in
+ * tracking the connection state. If a callback is not of interest for
+ * an instance, it may be set to NULL and will as a consequence not be
+ * used for that instance.
+ */
+struct bt_conn_cb {
+ /** @brief A new connection has been established.
+ *
+ * This callback notifies the application of a new connection.
+ * In case the err parameter is non-zero it means that the
+ * connection establishment failed.
+ *
+ * @param conn New connection object.
+ * @param err HCI error. Zero for success, non-zero otherwise.
+ */
+ void (*connected)(struct bt_conn *conn, u8_t err);
+
+ /** @brief A connection has been disconnected.
+ *
+ * This callback notifies the application that a connection
+ * has been disconnected.
+ *
+ * @param conn Connection object.
+ * @param reason HCI reason for the disconnection.
+ */
+ void (*disconnected)(struct bt_conn *conn, u8_t reason);
+
+ /** @brief LE connection parameter update request.
+ *
+ * This callback notifies the application that a remote device
+ * is requesting to update the connection parameters. The
+ * application accepts the parameters by returning true, or
+ * rejects them by returning false. Before accepting, the
+ * application may also adjust the parameters to better suit
+ * its needs.
+ *
+ * It is recommended for an application to have just one of these
+ * callbacks for simplicity. However, if an application registers
+ * multiple it needs to manage the potentially different
+ * requirements for each callback. Each callback gets the
+ * parameters as returned by previous callbacks, i.e. they are not
+ * necessarily the same ones as the remote originally sent.
+ *
+ * @param conn Connection object.
+ * @param param Proposed connection parameters.
+ *
+ * @return true to accept the parameters, or false to reject them.
+ */
+ bool (*le_param_req)(struct bt_conn *conn,
+ struct bt_le_conn_param *param);
+
+ /** @brief The parameters for an LE connection have been updated.
+ *
+ * This callback notifies the application that the connection
+ * parameters for an LE connection have been updated.
+ *
+ * @param conn Connection object.
+ * @param interval Connection interval.
+ * @param latency Connection latency.
+ * @param timeout Connection supervision timeout.
+ */
+ void (*le_param_updated)(struct bt_conn *conn, u16_t interval,
+ u16_t latency, u16_t timeout);
+#if defined(CONFIG_BT_SMP)
+ /** @brief Remote Identity Address has been resolved.
+ *
+ * This callback notifies the application that a remote
+ * Identity Address has been resolved
+ *
+ * @param conn Connection object.
+ * @param rpa Resolvable Private Address.
+ * @param identity Identity Address.
+ */
+ void (*identity_resolved)(struct bt_conn *conn,
+ const bt_addr_le_t *rpa,
+ const bt_addr_le_t *identity);
+#endif /* CONFIG_BT_SMP */
+#if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR)
+ /** @brief The security level of a connection has changed.
+ *
+ * This callback notifies the application that the security level
+ * of a connection has changed.
+ *
+ * @param conn Connection object.
+ * @param level New security level of the connection.
+ */
+ void (*security_changed)(struct bt_conn *conn, bt_security_t level);
+#endif /* defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) */
+ struct bt_conn_cb *_next;
+};
+
+#if CONFIG_BT_MESH_PROVISIONER
+struct bt_prov_conn_cb {
+ void (*connected)(const u8_t addr[6], struct bt_conn *conn, int id);
+
+ void (*disconnected)(struct bt_conn *conn, u8_t reason);
+
+ ssize_t (*prov_write_descr)(struct bt_conn *conn, u8_t *addr);
+
+ ssize_t (*prov_notify)(struct bt_conn *conn, u8_t *data, u16_t len);
+
+ ssize_t (*proxy_write_descr)(struct bt_conn *conn);
+
+ ssize_t (*proxy_notify)(struct bt_conn *conn, u8_t *data, u16_t len);
+
+ bool (*le_param_req)(struct bt_conn *conn, struct bt_le_conn_param *param);
+
+ void (*le_param_updated)(struct bt_conn *conn, u16_t interval,
+ u16_t latency, u16_t timeout);
+
+#if defined(CONFIG_BT_SMP)
+ void (*identity_resolved)(struct bt_conn *conn,
+ const bt_addr_le_t *rpa,
+ const bt_addr_le_t *identity);
+#endif /* CONFIG_BT_SMP */
+
+#if defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR)
+ void (*security_changed)(struct bt_conn *conn, bt_security_t level);
+#endif /* defined(CONFIG_BT_SMP) || defined(CONFIG_BT_BREDR) */
+
+ struct bt_conn_cb *_next;
+};
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+/** @brief GATT Attribute structure. */
+struct bt_gatt_attr {
+ /** Attribute UUID */
+ const struct bt_uuid *uuid;
+
+ /** Attribute read callback
+ *
+ * @param conn The connection that is requesting to read
+ * @param attr The attribute that's being read
+ * @param buf Buffer to place the read result in
+ * @param len Length of data to read
+ * @param offset Offset to start reading from
+ *
+ * @return Number fo bytes read, or in case of an error
+ * BT_GATT_ERR() with a specific ATT error code.
+ */
+ ssize_t (*read)(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ void *buf, u16_t len,
+ u16_t offset);
+
+ /** Attribute write callback
+ *
+ * @param conn The connection that is requesting to write
+ * @param attr The attribute that's being written
+ * @param buf Buffer with the data to write
+ * @param len Number of bytes in the buffer
+ * @param offset Offset to start writing from
+ * @param flags Flags (BT_GATT_WRITE_*)
+ *
+ * @return Number of bytes written, or in case of an error
+ * BT_GATT_ERR() with a specific ATT error code.
+ */
+ ssize_t (*write)(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ const void *buf, u16_t len,
+ u16_t offset, u8_t flags);
+
+ /** Attribute user data */
+ void *user_data;
+ /** Attribute handle */
+ u16_t handle;
+ /** Attribute permissions */
+ u8_t perm;
+};
+
+/** @def BT_GATT_PRIMARY_SERVICE
+ * @brief Primary Service Declaration Macro.
+ *
+ * Helper macro to declare a primary service attribute.
+ *
+ * @param _service Service attribute value.
+ */
+#define BT_GATT_PRIMARY_SERVICE(_service) \
+{ \
+ .uuid = BT_UUID_GATT_PRIMARY, \
+ .perm = BT_GATT_PERM_READ, \
+ .read = bt_mesh_gatts_attr_read_service, \
+ .user_data = _service, \
+}
+
+/** @def BT_GATT_SECONDARY_SERVICE
+ * @brief Secondary Service Declaration Macro.
+ *
+ * Helper macro to declare a secondary service attribute.
+ *
+ * @param _service Service attribute value.
+ */
+#define BT_GATT_SECONDARY_SERVICE(_service) \
+{ \
+ .uuid = BT_UUID_GATT_SECONDARY, \
+ .perm = BT_GATT_PERM_READ, \
+ .read = bt_mesh_gatts_attr_read_service, \
+ .user_data = _service, \
+}
+
+/** @def BT_GATT_INCLUDE_SERVICE
+ * @brief Include Service Declaration Macro.
+ *
+ * Helper macro to declare database internal include service attribute.
+ *
+ * @param _service_incl the first service attribute of service to include
+ */
+#define BT_GATT_INCLUDE_SERVICE(_service_incl) \
+{ \
+ .uuid = BT_UUID_GATT_INCLUDE, \
+ .perm = BT_GATT_PERM_READ, \
+ .read = bt_mesh_gatts_attr_read_included, \
+ .user_data = _service_incl, \
+}
+
+
+/** @def BT_GATT_CHARACTERISTIC
+ * @brief Characteristic Declaration Macro.
+ *
+ * Helper macro to declare a characteristic attribute.
+ *
+ * @param _uuid Characteristic attribute uuid.
+ * @param _props Characteristic attribute properties.
+ */
+#define BT_GATT_CHARACTERISTIC(_uuid, _props) \
+{ \
+ .uuid = BT_UUID_GATT_CHRC, \
+ .perm = BT_GATT_PERM_READ, \
+ .read = bt_mesh_gatts_attr_read_chrc, \
+ .user_data = (&(struct bt_gatt_chrc) { .uuid = _uuid, \
+ .properties = _props, }),\
+}
+
+/** @def BT_GATT_DESCRIPTOR
+ * @brief Descriptor Declaration Macro.
+ *
+ * Helper macro to declare a descriptor attribute.
+ *
+ * @param _uuid Descriptor attribute uuid.
+ * @param _perm Descriptor attribute access permissions.
+ * @param _read Descriptor attribute read callback.
+ * @param _write Descriptor attribute write callback.
+ * @param _value Descriptor attribute value.
+ */
+#define BT_GATT_DESCRIPTOR(_uuid, _perm, _read, _write, _value) \
+{ \
+ .uuid = _uuid, \
+ .perm = _perm, \
+ .read = _read, \
+ .write = _write, \
+ .user_data = _value, \
+}
+
+/** @def BT_GATT_SERVICE
+ * @brief Service Structure Declaration Macro.
+ *
+ * Helper macro to declare a service structure.
+ *
+ * @param _attrs Service attributes.
+ */
+#define BT_GATT_SERVICE(_attrs) \
+{ \
+ .attrs = _attrs, \
+ .attr_count = ARRAY_SIZE(_attrs), \
+}
+
+/* @brief Container for public key callback */
+struct bt_pub_key_cb {
+ /** @brief Callback type for Public Key generation.
+ *
+ * Used to notify of the local public key or that the local key is not
+ * available (either because of a failure to read it or because it is
+ * being regenerated).
+ *
+ * @param key The local public key, or NULL in case of no key.
+ */
+ void (*func)(const u8_t key[64]);
+
+ struct bt_pub_key_cb *_next;
+};
+
+int bt_le_adv_start(const struct bt_le_adv_param *param,
+ const struct bt_data *ad, size_t ad_len,
+ const struct bt_data *sd, size_t sd_len);
+
+int bt_le_adv_stop(void);
+
+int bt_le_scan_start(const struct bt_le_scan_param *param, bt_le_scan_cb_t cb);
+
+int bt_le_scan_stop(void);
+
+void bt_mesh_gatts_conn_cb_register(struct bt_conn_cb *cb);
+
+int bt_mesh_gatts_disconnect(struct bt_conn *conn, u8_t reason);
+
+int bt_mesh_gatts_service_register(struct bt_gatt_service *svc);
+
+int bt_mesh_gatts_service_unregister(struct bt_gatt_service *svc);
+
+ssize_t bt_mesh_gatts_attr_read_included(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ void *buf, u16_t len, u16_t offset);
+
+ssize_t bt_mesh_gatts_attr_read(struct bt_conn *conn, const struct bt_gatt_attr *attr,
+ void *buf, u16_t buf_len, u16_t offset,
+ const void *value, u16_t value_len);
+
+ssize_t bt_mesh_gatts_attr_read_service(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ void *buf, u16_t len, u16_t offset);
+
+ssize_t bt_mesh_gatts_attr_read_chrc(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr, void *buf,
+ u16_t len, u16_t offset);
+
+int bt_mesh_gatts_notify(struct bt_conn *conn, const struct bt_gatt_attr *attr,
+ const void *data, u16_t len);
+
+u16_t bt_mesh_gatt_get_mtu(struct bt_conn *conn);
+
+/** APIs added by Espressif */
+int bt_mesh_gatts_service_stop(struct bt_gatt_service *svc);
+int bt_mesh_gatts_service_start(struct bt_gatt_service *svc);
+
+#if CONFIG_BT_MESH_PROVISIONER
+void bt_mesh_gattc_conn_cb_register(struct bt_prov_conn_cb *cb);
+
+u16_t bt_mesh_gattc_get_service_uuid(struct bt_conn *conn);
+
+int bt_mesh_gattc_conn_create(const bt_addr_le_t *addr, u16_t service_uuid);
+
+void bt_gattc_conn_close(struct bt_conn *conn);
+
+void bt_mesh_gattc_exchange_mtu(u8_t index);
+
+u16_t bt_mesh_gattc_get_mtu_info(struct bt_conn *conn);
+
+int bt_mesh_gattc_write_no_rsp(struct bt_conn *conn, const struct bt_gatt_attr *attr,
+ const void *data, u16_t len);
+
+void bt_mesh_gattc_disconnect(struct bt_conn *conn);
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+struct bt_conn *bt_mesh_conn_ref(struct bt_conn *conn);
+
+void bt_mesh_conn_unref(struct bt_conn *conn);
+
+void bt_mesh_gatt_init(void);
+
+void bt_mesh_adapt_init(void);
+
+int bt_mesh_rand(void *buf, size_t len);
+
+void bt_mesh_set_private_key(const u8_t pri_key[32]);
+
+const u8_t *bt_mesh_pub_key_get(void);
+
+bool bt_mesh_check_public_key(const uint8_t key[64]);
+
+int bt_mesh_dh_key_gen(const u8_t remote_pk[64], bt_dh_key_cb_t cb);
+
+int bt_encrypt_le(const u8_t key[16], const u8_t plaintext[16],
+ u8_t enc_data[16]);
+
+int bt_encrypt_be(const u8_t key[16], const u8_t plaintext[16],
+ u8_t enc_data[16]);
+
+#endif /* #ifndef __MESH_BEARER_ADRPT_H__ */
+
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_bt_uuid.h b/components/bt/ble_mesh/mesh_core/include/mesh_bt_uuid.h
new file mode 100644
index 0000000000..80f52023fc
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_bt_uuid.h
@@ -0,0 +1,580 @@
+/** @file
+ * @brief Bluetooth UUID handling
+ */
+
+/*
+ * Copyright (c) 2015-2016 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __BT_UUID_H
+#define __BT_UUID_H
+
+/**
+ * @brief UUIDs
+ * @defgroup bt_uuid UUIDs
+ * @ingroup bluetooth
+ * @{
+ */
+
+#include "mesh_util.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/** @brief Bluetooth UUID types */
+enum {
+ BT_UUID_TYPE_16,
+ BT_UUID_TYPE_32,
+ BT_UUID_TYPE_128,
+};
+
+/** @brief This is a 'tentative' type and should be used as a pointer only */
+struct bt_uuid {
+ u8_t type;
+};
+
+struct bt_uuid_16 {
+ struct bt_uuid uuid;
+ u16_t val;
+};
+
+struct bt_uuid_32 {
+ struct bt_uuid uuid;
+ u32_t val;
+};
+
+struct bt_uuid_128 {
+ struct bt_uuid uuid;
+ u8_t val[16];
+};
+
+#define BT_UUID_INIT_16(value) \
+{ \
+ .uuid.type = BT_UUID_TYPE_16, \
+ .val = (value), \
+}
+
+#define BT_UUID_INIT_32(value) \
+{ \
+ .uuid.type = BT_UUID_TYPE_32, \
+ .val = (value), \
+}
+
+#define BT_UUID_INIT_128(value...) \
+{ \
+ .uuid.type = BT_UUID_TYPE_128, \
+ .val = { value }, \
+}
+
+#define BT_UUID_DECLARE_16(value) \
+ ((struct bt_uuid *) (&(struct bt_uuid_16) BT_UUID_INIT_16(value)))
+#define BT_UUID_DECLARE_32(value) \
+ ((struct bt_uuid *) (&(struct bt_uuid_32) BT_UUID_INIT_32(value)))
+#define BT_UUID_DECLARE_128(value...) \
+ ((struct bt_uuid *) (&(struct bt_uuid_128) BT_UUID_INIT_128(value)))
+
+#define BT_UUID_16(__u) CONTAINER_OF(__u, struct bt_uuid_16, uuid)
+#define BT_UUID_32(__u) CONTAINER_OF(__u, struct bt_uuid_32, uuid)
+#define BT_UUID_128(__u) CONTAINER_OF(__u, struct bt_uuid_128, uuid)
+
+/** @def BT_UUID_GAP
+ * @brief Generic Access
+ */
+#define BT_UUID_GAP BT_UUID_DECLARE_16(0x1800)
+#define BT_UUID_GAP_VAL 0x1800
+/** @def BT_UUID_GATT
+ * @brief Generic Attribute
+ */
+#define BT_UUID_GATT BT_UUID_DECLARE_16(0x1801)
+#define BT_UUID_GATT_VAL 0x1801
+/** @def BT_UUID_CTS
+ * @brief Current Time Service
+ */
+#define BT_UUID_CTS BT_UUID_DECLARE_16(0x1805)
+#define BT_UUID_CTS_VAL 0x1805
+/** @def BT_UUID_DIS
+ * @brief Device Information Service
+ */
+#define BT_UUID_DIS BT_UUID_DECLARE_16(0x180a)
+#define BT_UUID_DIS_VAL 0x180a
+/** @def BT_UUID_HRS
+ * @brief Heart Rate Service
+ */
+#define BT_UUID_HRS BT_UUID_DECLARE_16(0x180d)
+#define BT_UUID_HRS_VAL 0x180d
+/** @def BT_UUID_BAS
+ * @brief Battery Service
+ */
+#define BT_UUID_BAS BT_UUID_DECLARE_16(0x180f)
+#define BT_UUID_BAS_VAL 0x180f
+/** @def BT_UUID_HIDS
+ * @brief HID Service
+ */
+#define BT_UUID_HIDS BT_UUID_DECLARE_16(0x1812)
+#define BT_UUID_HIDS_VAL 0x1812
+/** @def BT_UUID_CSC
+ * @brief Cycling Speed and Cadence Service
+ */
+#define BT_UUID_CSC BT_UUID_DECLARE_16(0x1816)
+#define BT_UUID_CSC_VAL 0x1816
+/** @def BT_UUID_ESS
+ * @brief Environmental Sensing Service
+ */
+#define BT_UUID_ESS BT_UUID_DECLARE_16(0x181a)
+#define BT_UUID_ESS_VAL 0x181a
+/** @def BT_UUID_IPSS
+ * @brief IP Support Service
+ */
+#define BT_UUID_IPSS BT_UUID_DECLARE_16(0x1820)
+#define BT_UUID_IPSS_VAL 0x1820
+/** @def BT_UUID_MESH_PROV
+ * @brief Mesh Provisioning Service
+ */
+#define BT_UUID_MESH_PROV BT_UUID_DECLARE_16(0x1827)
+#define BT_UUID_MESH_PROV_VAL 0x1827
+/** @def BT_UUID_MESH_PROXY
+ * @brief Mesh Proxy Service
+ */
+#define BT_UUID_MESH_PROXY BT_UUID_DECLARE_16(0x1828)
+#define BT_UUID_MESH_PROXY_VAL 0x1828
+/** @def BT_UUID_GATT_PRIMARY
+ * @brief GATT Primary Service
+ */
+#define BT_UUID_GATT_PRIMARY BT_UUID_DECLARE_16(0x2800)
+#define BT_UUID_GATT_PRIMARY_VAL 0x2800
+/** @def BT_UUID_GATT_SECONDARY
+ * @brief GATT Secondary Service
+ */
+#define BT_UUID_GATT_SECONDARY BT_UUID_DECLARE_16(0x2801)
+#define BT_UUID_GATT_SECONDARY_VAL 0x2801
+/** @def BT_UUID_GATT_INCLUDE
+ * @brief GATT Include Service
+ */
+#define BT_UUID_GATT_INCLUDE BT_UUID_DECLARE_16(0x2802)
+#define BT_UUID_GATT_INCLUDE_VAL 0x2802
+/** @def BT_UUID_GATT_CHRC
+ * @brief GATT Characteristic
+ */
+#define BT_UUID_GATT_CHRC BT_UUID_DECLARE_16(0x2803)
+#define BT_UUID_GATT_CHRC_VAL 0x2803
+/** @def BT_UUID_GATT_CEP
+ * @brief GATT Characteristic Extended Properties
+ */
+#define BT_UUID_GATT_CEP BT_UUID_DECLARE_16(0x2900)
+#define BT_UUID_GATT_CEP_VAL 0x2900
+/** @def BT_UUID_GATT_CUD
+ * @brief GATT Characteristic User Description
+ */
+#define BT_UUID_GATT_CUD BT_UUID_DECLARE_16(0x2901)
+#define BT_UUID_GATT_CUD_VAL 0x2901
+/** @def BT_UUID_GATT_CCC
+ * @brief GATT Client Characteristic Configuration
+ */
+#define BT_UUID_GATT_CCC BT_UUID_DECLARE_16(0x2902)
+#define BT_UUID_GATT_CCC_VAL 0x2902
+/** @def BT_UUID_GATT_SCC
+ * @brief GATT Server Characteristic Configuration
+ */
+#define BT_UUID_GATT_SCC BT_UUID_DECLARE_16(0x2903)
+#define BT_UUID_GATT_SCC_VAL 0x2903
+/** @def BT_UUID_GATT_CPF
+ * @brief GATT Characteristic Presentation Format
+ */
+#define BT_UUID_GATT_CPF BT_UUID_DECLARE_16(0x2904)
+#define BT_UUID_GATT_CPF_VAL 0x2904
+/** @def BT_UUID_VALID_RANGE
+ * @brief Valid Range Descriptor
+ */
+#define BT_UUID_VALID_RANGE BT_UUID_DECLARE_16(0x2906)
+#define BT_UUID_VALID_RANGE_VAL 0x2906
+/** @def BT_UUID_HIDS_EXT_REPORT
+ * @brief HID External Report Descriptor
+ */
+#define BT_UUID_HIDS_EXT_REPORT BT_UUID_DECLARE_16(0x2907)
+#define BT_UUID_HIDS_EXT_REPORT_VAL 0x2907
+/** @def BT_UUID_HIDS_REPORT_REF
+ * @brief HID Report Reference Descriptor
+ */
+#define BT_UUID_HIDS_REPORT_REF BT_UUID_DECLARE_16(0x2908)
+#define BT_UUID_HIDS_REPORT_REF_VAL 0x2908
+/** @def BT_UUID_ES_CONFIGURATION
+ * @brief Environmental Sensing Configuration Descriptor
+ */
+#define BT_UUID_ES_CONFIGURATION BT_UUID_DECLARE_16(0x290b)
+#define BT_UUID_ES_CONFIGURATION_VAL 0x290b
+/** @def BT_UUID_ES_MEASUREMENT
+ * @brief Environmental Sensing Measurement Descriptor
+ */
+#define BT_UUID_ES_MEASUREMENT BT_UUID_DECLARE_16(0x290c)
+#define BT_UUID_ES_MEASUREMENT_VAL 0x290c
+/** @def BT_UUID_ES_TRIGGER_SETTING
+ * @brief Environmental Sensing Trigger Setting Descriptor
+ */
+#define BT_UUID_ES_TRIGGER_SETTING BT_UUID_DECLARE_16(0x290d)
+#define BT_UUID_ES_TRIGGER_SETTING_VAL 0x290d
+/** @def BT_UUID_GAP_DEVICE_NAME
+ * @brief GAP Characteristic Device Name
+ */
+#define BT_UUID_GAP_DEVICE_NAME BT_UUID_DECLARE_16(0x2a00)
+#define BT_UUID_GAP_DEVICE_NAME_VAL 0x2a00
+/** @def BT_UUID_GAP_APPEARANCE
+ * @brief GAP Characteristic Appearance
+ */
+#define BT_UUID_GAP_APPEARANCE BT_UUID_DECLARE_16(0x2a01)
+#define BT_UUID_GAP_APPEARANCE_VAL 0x2a01
+/** @def BT_UUID_GAP_PPCP
+ * @brief GAP Characteristic Peripheral Preferred Connection Parameters
+ */
+#define BT_UUID_GAP_PPCP BT_UUID_DECLARE_16(0x2a04)
+#define BT_UUID_GAP_PPCP_VAL 0x2a04
+/** @def BT_UUID_GATT_SC
+ * @brief GATT Characteristic Service Changed
+ */
+#define BT_UUID_GATT_SC BT_UUID_DECLARE_16(0x2a05)
+#define BT_UUID_GATT_SC_VAL 0x2a05
+/** @def BT_UUID_BAS_BATTERY_LEVEL
+ * @brief BAS Characteristic Battery Level
+ */
+#define BT_UUID_BAS_BATTERY_LEVEL BT_UUID_DECLARE_16(0x2a19)
+#define BT_UUID_BAS_BATTERY_LEVEL_VAL 0x2a19
+/** @def BT_UUID_DIS_SYSTEM_ID
+ * @brief DIS Characteristic System ID
+ */
+#define BT_UUID_DIS_SYSTEM_ID BT_UUID_DECLARE_16(0x2a23)
+#define BT_UUID_DIS_SYSTEM_ID_VAL 0x2a23
+/** @def BT_UUID_DIS_MODEL_NUMBER
+ * @brief DIS Characteristic Model Number String
+ */
+#define BT_UUID_DIS_MODEL_NUMBER BT_UUID_DECLARE_16(0x2a24)
+#define BT_UUID_DIS_MODEL_NUMBER_VAL 0x2a24
+/** @def BT_UUID_DIS_SERIAL_NUMBER
+ * @brief DIS Characteristic Serial Number String
+ */
+#define BT_UUID_DIS_SERIAL_NUMBER BT_UUID_DECLARE_16(0x2a25)
+#define BT_UUID_DIS_SERIAL_NUMBER_VAL 0x2a25
+/** @def BT_UUID_DIS_FIRMWARE_REVISION
+ * @brief DIS Characteristic Firmware Revision String
+ */
+#define BT_UUID_DIS_FIRMWARE_REVISION BT_UUID_DECLARE_16(0x2a26)
+#define BT_UUID_DIS_FIRMWARE_REVISION_VAL 0x2a26
+/** @def BT_UUID_DIS_HARDWARE_REVISION
+ * @brief DIS Characteristic Hardware Revision String
+ */
+#define BT_UUID_DIS_HARDWARE_REVISION BT_UUID_DECLARE_16(0x2a27)
+#define BT_UUID_DIS_HARDWARE_REVISION_VAL 0x2a27
+/** @def BT_UUID_DIS_SOFTWARE_REVISION
+ * @brief DIS Characteristic Software Revision String
+ */
+#define BT_UUID_DIS_SOFTWARE_REVISION BT_UUID_DECLARE_16(0x2a28)
+#define BT_UUID_DIS_SOFTWARE_REVISION_VAL 0x2a28
+/** @def BT_UUID_DIS_MANUFACTURER_NAME
+ * @brief DIS Characteristic Manufacturer Name String
+ */
+#define BT_UUID_DIS_MANUFACTURER_NAME BT_UUID_DECLARE_16(0x2a29)
+#define BT_UUID_DIS_MANUFACTURER_NAME_VAL 0x2a29
+/** @def BT_UUID_DIS_PNP_ID
+ * @brief DIS Characteristic PnP ID
+ */
+#define BT_UUID_DIS_PNP_ID BT_UUID_DECLARE_16(0x2a50)
+#define BT_UUID_DIS_PNP_ID_VAL 0x2a50
+/** @def BT_UUID_CTS_CURRENT_TIME
+ * @brief CTS Characteristic Current Time
+ */
+#define BT_UUID_CTS_CURRENT_TIME BT_UUID_DECLARE_16(0x2a2b)
+#define BT_UUID_CTS_CURRENT_TIME_VAL 0x2a2b
+/** @def BT_UUID_MAGN_DECLINATION
+ * @brief Magnetic Declination Characteristic
+ */
+#define BT_UUID_MAGN_DECLINATION BT_UUID_DECLARE_16(0x2a2c)
+#define BT_UUID_MAGN_DECLINATION_VAL 0x2a2c
+/** @def BT_UUID_HRS_MEASUREMENT
+ * @brief HRS Characteristic Measurement Interval
+ */
+#define BT_UUID_HRS_MEASUREMENT BT_UUID_DECLARE_16(0x2a37)
+#define BT_UUID_HRS_MEASUREMENT_VAL 0x2a37
+/** @def BT_UUID_HRS_BODY_SENSOR
+ * @brief HRS Characteristic Body Sensor Location
+ */
+#define BT_UUID_HRS_BODY_SENSOR BT_UUID_DECLARE_16(0x2a38)
+#define BT_UUID_HRS_BODY_SENSOR_VAL 0x2a38
+/** @def BT_UUID_HRS_CONTROL_POINT
+ * @brief HRS Characteristic Control Point
+ */
+#define BT_UUID_HRS_CONTROL_POINT BT_UUID_DECLARE_16(0x2a39)
+#define BT_UUID_HRS_CONTROL_POINT_VAL 0x2a39
+/** @def BT_UUID_HIDS_INFO
+ * @brief HID Information Characteristic
+ */
+#define BT_UUID_HIDS_INFO BT_UUID_DECLARE_16(0x2a4a)
+#define BT_UUID_HIDS_INFO_VAL 0x2a4a
+/** @def BT_UUID_HIDS_REPORT_MAP
+ * @brief HID Report Map Characteristic
+ */
+#define BT_UUID_HIDS_REPORT_MAP BT_UUID_DECLARE_16(0x2a4b)
+#define BT_UUID_HIDS_REPORT_MAP_VAL 0x2a4b
+/** @def BT_UUID_HIDS_CTRL_POINT
+ * @brief HID Control Point Characteristic
+ */
+#define BT_UUID_HIDS_CTRL_POINT BT_UUID_DECLARE_16(0x2a4c)
+#define BT_UUID_HIDS_CTRL_POINT_VAL 0x2a4c
+/** @def BT_UUID_HIDS_REPORT
+ * @brief HID Report Characteristic
+ */
+#define BT_UUID_HIDS_REPORT BT_UUID_DECLARE_16(0x2a4d)
+#define BT_UUID_HIDS_REPORT_VAL 0x2a4d
+/** @def BT_UUID_CSC_MEASUREMENT
+ * @brief CSC Measurement Characteristic
+ */
+#define BT_UUID_CSC_MEASUREMENT BT_UUID_DECLARE_16(0x2a5b)
+#define BT_UUID_CSC_MEASUREMENT_VAL 0x2a5b
+/** @def BT_UUID_CSC_FEATURE
+ * @brief CSC Feature Characteristic
+ */
+#define BT_UUID_CSC_FEATURE BT_UUID_DECLARE_16(0x2a5c)
+#define BT_UUID_CSC_FEATURE_VAL 0x2a5c
+/** @def BT_UUID_SENSOR_LOCATION
+ * @brief Sensor Location Characteristic
+ */
+#define BT_UUID_SENSOR_LOCATION BT_UUID_DECLARE_16(0x2a5d)
+#define BT_UUID_SENSOR_LOCATION_VAL 0x2a5d
+/** @def BT_UUID_SC_CONTROL_POINT
+ * @brief SC Control Point Characteristic
+ */
+#define BT_UUID_SC_CONTROL_POINT BT_UUID_DECLARE_16(0x2a55)
+#define BT_UUID_SC_CONTROL_POINT_VAl 0x2a55
+/** @def BT_UUID_ELEVATION
+ * @brief Elevation Characteristic
+ */
+#define BT_UUID_ELEVATION BT_UUID_DECLARE_16(0x2a6c)
+#define BT_UUID_ELEVATION_VAL 0x2a6c
+/** @def BT_UUID_PRESSURE
+ * @brief Pressure Characteristic
+ */
+#define BT_UUID_PRESSURE BT_UUID_DECLARE_16(0x2a6d)
+#define BT_UUID_PRESSURE_VAL 0x2a6d
+/** @def BT_UUID_TEMPERATURE
+ * @brief Temperature Characteristic
+ */
+#define BT_UUID_TEMPERATURE BT_UUID_DECLARE_16(0x2a6e)
+#define BT_UUID_TEMPERATURE_VAL 0x2a6e
+/** @def BT_UUID_HUMIDITY
+ * @brief Humidity Characteristic
+ */
+#define BT_UUID_HUMIDITY BT_UUID_DECLARE_16(0x2a6f)
+#define BT_UUID_HUMIDITY_VAL 0x2a6f
+/** @def BT_UUID_TRUE_WIND_SPEED
+ * @brief True Wind Speed Characteristic
+ */
+#define BT_UUID_TRUE_WIND_SPEED BT_UUID_DECLARE_16(0x2a70)
+#define BT_UUID_TRUE_WIND_SPEED_VAL 0x2a70
+/** @def BT_UUID_TRUE_WIND_DIR
+ * @brief True Wind Direction Characteristic
+ */
+#define BT_UUID_TRUE_WIND_DIR BT_UUID_DECLARE_16(0x2a71)
+#define BT_UUID_TRUE_WIND_DIR_VAL 0x2a71
+/** @def BT_UUID_APPARENT_WIND_SPEED
+ * @brief Apparent Wind Speed Characteristic
+ */
+#define BT_UUID_APPARENT_WIND_SPEED BT_UUID_DECLARE_16(0x2a72)
+#define BT_UUID_APPARENT_WIND_SPEED_VAL 0x2a72
+/** @def BT_UUID_APPARENT_WIND_DIR
+ * @brief Apparent Wind Direction Characteristic
+ */
+#define BT_UUID_APPARENT_WIND_DIR BT_UUID_DECLARE_16(0x2a73)
+#define BT_UUID_APPARENT_WIND_DIR_VAL 0x2a73
+/** @def BT_UUID_GUST_FACTOR
+ * @brief Gust Factor Characteristic
+ */
+#define BT_UUID_GUST_FACTOR BT_UUID_DECLARE_16(0x2a74)
+#define BT_UUID_GUST_FACTOR_VAL 0x2a74
+/** @def BT_UUID_POLLEN_CONCENTRATION
+ * @brief Pollen Concentration Characteristic
+ */
+#define BT_UUID_POLLEN_CONCENTRATION BT_UUID_DECLARE_16(0x2a75)
+#define BT_UUID_POLLEN_CONCENTRATION_VAL 0x2a75
+/** @def BT_UUID_UV_INDEX
+ * @brief UV Index Characteristic
+ */
+#define BT_UUID_UV_INDEX BT_UUID_DECLARE_16(0x2a76)
+#define BT_UUID_UV_INDEX_VAL 0x2a76
+/** @def BT_UUID_IRRADIANCE
+ * @brief Irradiance Characteristic
+ */
+#define BT_UUID_IRRADIANCE BT_UUID_DECLARE_16(0x2a77)
+#define BT_UUID_IRRADIANCE_VAL 0x2a77
+/** @def BT_UUID_RAINFALL
+ * @brief Rainfall Characteristic
+ */
+#define BT_UUID_RAINFALL BT_UUID_DECLARE_16(0x2a78)
+#define BT_UUID_RAINFALL_VAL 0x2a78
+/** @def BT_UUID_WIND_CHILL
+ * @brief Wind Chill Characteristic
+ */
+#define BT_UUID_WIND_CHILL BT_UUID_DECLARE_16(0x2a79)
+#define BT_UUID_WIND_CHILL_VAL 0x2a79
+/** @def BT_UUID_HEAT_INDEX
+ * @brief Heat Index Characteristic
+ */
+#define BT_UUID_HEAT_INDEX BT_UUID_DECLARE_16(0x2a7a)
+#define BT_UUID_HEAT_INDEX_VAL 0x2a7a
+/** @def BT_UUID_DEW_POINT
+ * @brief Dew Point Characteristic
+ */
+#define BT_UUID_DEW_POINT BT_UUID_DECLARE_16(0x2a7b)
+#define BT_UUID_DEW_POINT_VAL 0x2a7b
+/** @def BT_UUID_DESC_VALUE_CHANGED
+ * @brief Descriptor Value Changed Characteristic
+ */
+#define BT_UUID_DESC_VALUE_CHANGED BT_UUID_DECLARE_16(0x2a7d)
+#define BT_UUID_DESC_VALUE_CHANGED_VAL 0x2a7d
+/** @def BT_UUID_MAGN_FLUX_DENSITY_2D
+ * @brief Magnetic Flux Density - 2D Characteristic
+ */
+#define BT_UUID_MAGN_FLUX_DENSITY_2D BT_UUID_DECLARE_16(0x2aa0)
+#define BT_UUID_MAGN_FLUX_DENSITY_2D_VAL 0x2aa0
+/** @def BT_UUID_MAGN_FLUX_DENSITY_3D
+ * @brief Magnetic Flux Density - 3D Characteristic
+ */
+#define BT_UUID_MAGN_FLUX_DENSITY_3D BT_UUID_DECLARE_16(0x2aa1)
+#define BT_UUID_MAGN_FLUX_DENSITY_3D_VAL 0x2aa1
+/** @def BT_UUID_BAR_PRESSURE_TREND
+ * @brief Barometric Pressure Trend Characteristic
+ */
+#define BT_UUID_BAR_PRESSURE_TREND BT_UUID_DECLARE_16(0x2aa3)
+#define BT_UUID_BAR_PRESSURE_TREND_VAL 0x2aa3
+/** @def BT_UUID_MESH_PROV_DATA_IN
+ * @brief Mesh Provisioning Data In
+ */
+#define BT_UUID_MESH_PROV_DATA_IN BT_UUID_DECLARE_16(0x2adb)
+#define BT_UUID_MESH_PROV_DATA_IN_VAL 0x2adb
+/** @def BT_UUID_MESH_PROV_DATA_OUT
+ * @brief Mesh Provisioning Data Out
+ */
+#define BT_UUID_MESH_PROV_DATA_OUT BT_UUID_DECLARE_16(0x2adc)
+#define BT_UUID_MESH_PROV_DATA_OUT_VAL 0x2adc
+/** @def BT_UUID_MESH_PROXY_DATA_IN
+ * @brief Mesh Proxy Data In
+ */
+#define BT_UUID_MESH_PROXY_DATA_IN BT_UUID_DECLARE_16(0x2add)
+#define BT_UUID_MESH_PROXY_DATA_IN_VAL 0x2add
+/** @def BT_UUID_MESH_PROXY_DATA_OUT
+ * @brief Mesh Proxy Data Out
+ */
+#define BT_UUID_MESH_PROXY_DATA_OUT BT_UUID_DECLARE_16(0x2ade)
+#define BT_UUID_MESH_PROXY_DATA_OUT_VAL 0x2ade
+
+/*
+ * Protocol UUIDs
+ */
+#define BT_UUID_SDP BT_UUID_DECLARE_16(0x0001)
+#define BT_UUID_SDP_VAL 0x0001
+#define BT_UUID_UDP BT_UUID_DECLARE_16(0x0002)
+#define BT_UUID_UDP_VAL 0x0002
+#define BT_UUID_RFCOMM BT_UUID_DECLARE_16(0x0003)
+#define BT_UUID_RFCOMM_VAL 0x0003
+#define BT_UUID_TCP BT_UUID_DECLARE_16(0x0004)
+#define BT_UUID_TCP_VAL 0x0004
+#define BT_UUID_TCS_BIN BT_UUID_DECLARE_16(0x0005)
+#define BT_UUID_TCS_BIN_VAL 0x0005
+#define BT_UUID_TCS_AT BT_UUID_DECLARE_16(0x0006)
+#define BT_UUID_TCS_AT_VAL 0x0006
+#define BT_UUID_ATT BT_UUID_DECLARE_16(0x0007)
+#define BT_UUID_ATT_VAL 0x0007
+#define BT_UUID_OBEX BT_UUID_DECLARE_16(0x0008)
+#define BT_UUID_OBEX_VAL 0x0008
+#define BT_UUID_IP BT_UUID_DECLARE_16(0x0009)
+#define BT_UUID_IP_VAL 0x0009
+#define BT_UUID_FTP BT_UUID_DECLARE_16(0x000a)
+#define BT_UUID_FTP_VAL 0x000a
+#define BT_UUID_HTTP BT_UUID_DECLARE_16(0x000c)
+#define BT_UUID_HTTP_VAL 0x000c
+#define BT_UUID_BNEP BT_UUID_DECLARE_16(0x000f)
+#define BT_UUID_BNEP_VAL 0x000f
+#define BT_UUID_UPNP BT_UUID_DECLARE_16(0x0010)
+#define BT_UUID_UPNP_VAL 0x0010
+#define BT_UUID_HIDP BT_UUID_DECLARE_16(0x0011)
+#define BT_UUID_HIDP_VAL 0x0011
+#define BT_UUID_HCRP_CTRL BT_UUID_DECLARE_16(0x0012)
+#define BT_UUID_HCRP_CTRL_VAL 0x0012
+#define BT_UUID_HCRP_DATA BT_UUID_DECLARE_16(0x0014)
+#define BT_UUID_HCRP_DATA_VAL 0x0014
+#define BT_UUID_HCRP_NOTE BT_UUID_DECLARE_16(0x0016)
+#define BT_UUID_HCRP_NOTE_VAL 0x0016
+#define BT_UUID_AVCTP BT_UUID_DECLARE_16(0x0017)
+#define BT_UUID_AVCTP_VAL 0x0017
+#define BT_UUID_AVDTP BT_UUID_DECLARE_16(0x0019)
+#define BT_UUID_AVDTP_VAL 0x0019
+#define BT_UUID_CMTP BT_UUID_DECLARE_16(0x001b)
+#define BT_UUID_CMTP_VAL 0x001b
+#define BT_UUID_UDI BT_UUID_DECLARE_16(0x001d)
+#define BT_UUID_UDI_VAL 0x001d
+#define BT_UUID_MCAP_CTRL BT_UUID_DECLARE_16(0x001e)
+#define BT_UUID_MCAP_CTRL_VAL 0x001e
+#define BT_UUID_MCAP_DATA BT_UUID_DECLARE_16(0x001f)
+#define BT_UUID_MCAP_DATA_VAL 0x001f
+#define BT_UUID_L2CAP BT_UUID_DECLARE_16(0x0100)
+#define BT_UUID_L2CAP_VAL 0x0100
+
+
+/** @brief Compare Bluetooth UUIDs.
+ *
+ * Compares 2 Bluetooth UUIDs, if the types are different both UUIDs are
+ * first converted to 128 bits format before comparing.
+ *
+ * @param u1 First Bluetooth UUID to compare
+ * @param u2 Second Bluetooth UUID to compare
+ *
+ * @return negative value if @a u1 < @a u2, 0 if @a u1 == @a u2, else positive
+ */
+int bt_uuid_cmp(const struct bt_uuid *u1, const struct bt_uuid *u2);
+
+#if defined(CONFIG_BT_DEBUG)
+/** @brief Convert Bluetooth UUID to string.
+ *
+ * Converts Bluetooth UUID to string. UUID has to be in 16 bits or 128 bits
+ * format.
+ *
+ * @param uuid Bluetooth UUID
+ * @param str pointer where to put converted string
+ * @param len length of str
+ *
+ * @return N/A
+ */
+void bt_uuid_to_str(const struct bt_uuid *uuid, char *str, size_t len);
+
+/** @brief Convert Bluetooth UUID to string in place.
+ *
+ * Converts Bluetooth UUID to string in place. UUID has to be in 16 bits or
+ * 128 bits format.
+ *
+ * @param uuid Bluetooth UUID
+ *
+ * @return String representation of the UUID given
+ */
+const char *bt_uuid_str(const struct bt_uuid *uuid);
+#else
+static inline void bt_uuid_to_str(const struct bt_uuid *uuid, char *str,
+ size_t len)
+{
+ if (len > 0) {
+ str[0] = '\0';
+ }
+}
+
+static inline const char *bt_uuid_str(const struct bt_uuid *uuid)
+{
+ return "";
+}
+#endif /* CONFIG_BT_DEBUG */
+
+#ifdef __cplusplus
+}
+#endif
+
+/**
+ * @}
+ */
+
+#endif /* __BT_UUID_H */
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_buf.h b/components/bt/ble_mesh/mesh_core/include/mesh_buf.h
new file mode 100644
index 0000000000..4dbb83316c
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_buf.h
@@ -0,0 +1,1118 @@
+/** @file
+ * @brief Buffer management.
+ */
+
+/*
+ * Copyright (c) 2015 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __NET_BUF_H
+#define __NET_BUF_H
+
+#include
+#include "sys/cdefs.h"
+#include "mesh_types.h"
+#include "mesh_slist.h"
+#include "mesh_kernel.h"
+#include "mesh_util.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+/* Unaligned access */
+#define UNALIGNED_GET(p) \
+__extension__ ({ \
+ struct __attribute__((__packed__)) { \
+ __typeof__(*(p)) __v; \
+ } *__p = (__typeof__(__p)) (p); \
+ __p->__v; \
+})
+
+/**
+ * @brief Network buffer library
+ * @defgroup net_buf Network Buffer Library
+ * @ingroup networking
+ * @{
+ */
+
+/* Alignment needed for various parts of the buffer definition */
+#define __net_buf_align __aligned(sizeof(int))
+
+/** @def NET_BUF_SIMPLE
+ * @brief Define a net_buf_simple stack variable and get a pointer to it.
+ *
+ * This is a helper macro which is used to define a net_buf_simple object on
+ * the stack and the get a pointer to it as follows:
+ *
+ * struct net_buf_simple *my_buf = NET_BUF_SIMPLE(10);
+ *
+ * After creating the object it needs to be initialized by calling
+ * net_buf_simple_init().
+ *
+ * @param _size Maximum data storage for the buffer.
+ *
+ * @return Pointer to stack-allocated net_buf_simple object.
+ */
+#define NET_BUF_SIMPLE(_size) \
+ ((struct net_buf_simple *)(&(struct { \
+ struct net_buf_simple buf; \
+ u8_t data[_size] __net_buf_align; \
+ }) { \
+ .buf.size = _size, \
+ }))
+
+/** @brief Simple network buffer representation.
+ *
+ * This is a simpler variant of the net_buf object (in fact net_buf uses
+ * net_buf_simple internally). It doesn't provide any kind of reference
+ * counting, user data, dynamic allocation, or in general the ability to
+ * pass through kernel objects such as FIFOs.
+ *
+ * The main use of this is for scenarios where the meta-data of the normal
+ * net_buf isn't needed and causes too much overhead. This could be e.g.
+ * when the buffer only needs to be allocated on the stack or when the
+ * access to and lifetime of the buffer is well controlled and constrained.
+ *
+ */
+struct net_buf_simple {
+ /** Pointer to the start of data in the buffer. */
+ u8_t *data;
+
+ /** Length of the data behind the data pointer. */
+ u16_t len;
+
+ /** Amount of data that this buffer can store. */
+ u16_t size;
+
+ /** Start of the data storage. Not to be accessed directly
+ * (the data pointer should be used instead).
+ */
+ u8_t __buf[0] __net_buf_align;
+};
+
+/** @def NET_BUF_SIMPLE_DEFINE_STATIC
+ * @brief Define a static net_buf_simple variable.
+ *
+ * This is a helper macro which is used to define a static net_buf_simple
+ * object.
+ *
+ * @param _name Name of the net_buf_simple object.
+ * @param _size Maximum data storage for the buffer.
+ */
+#define NET_BUF_SIMPLE_DEFINE_STATIC(_name, _size) \
+ static __noinit u8_t net_buf_data_##_name[_size]; \
+ static struct net_buf_simple _name = { \
+ .data = net_buf_data_##_name, \
+ .len = 0, \
+ .size = _size, \
+ .__buf = net_buf_data_##_name, \
+ }
+
+/** @brief Initialize a net_buf_simple object.
+ *
+ * This needs to be called after creating a net_buf_simple object e.g. using
+ * the NET_BUF_SIMPLE macro.
+ *
+ * @param buf Buffer to initialize.
+ * @param reserve_head Headroom to reserve.
+ */
+static inline void net_buf_simple_init(struct net_buf_simple *buf,
+ size_t reserve_head)
+{
+ buf->data = buf->__buf + reserve_head;
+ buf->len = 0;
+}
+
+/**
+ * @brief Prepare data to be added at the end of the buffer
+ *
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param len Number of bytes to increment the length with.
+ *
+ * @return The original tail of the buffer.
+ */
+void *net_buf_simple_add(struct net_buf_simple *buf, size_t len);
+
+/**
+ * @brief Copy bytes from memory to the end of the buffer
+ *
+ * Copies the given number of bytes to the end of the buffer. Increments the
+ * data length of the buffer to account for more data at the end.
+ *
+ * @param buf Buffer to update.
+ * @param mem Location of data to be added.
+ * @param len Length of data to be added
+ *
+ * @return The original tail of the buffer.
+ */
+void *net_buf_simple_add_mem(struct net_buf_simple *buf, const void *mem,
+ size_t len);
+
+/**
+ * @brief Add (8-bit) byte at the end of the buffer
+ *
+ * Adds a byte at the end of the buffer. Increments the data length of
+ * the buffer to account for more data at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val byte value to be added.
+ *
+ * @return Pointer to the value added
+ */
+u8_t *net_buf_simple_add_u8(struct net_buf_simple *buf, u8_t val);
+
+/**
+ * @brief Add 16-bit value at the end of the buffer
+ *
+ * Adds 16-bit value in little endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be added.
+ */
+void net_buf_simple_add_le16(struct net_buf_simple *buf, u16_t val);
+
+/**
+ * @brief Add 16-bit value at the end of the buffer
+ *
+ * Adds 16-bit value in big endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be added.
+ */
+void net_buf_simple_add_be16(struct net_buf_simple *buf, u16_t val);
+
+/**
+ * @brief Add 32-bit value at the end of the buffer
+ *
+ * Adds 32-bit value in little endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 32-bit value to be added.
+ */
+void net_buf_simple_add_le32(struct net_buf_simple *buf, u32_t val);
+
+/**
+ * @brief Add 32-bit value at the end of the buffer
+ *
+ * Adds 32-bit value in big endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 32-bit value to be added.
+ */
+void net_buf_simple_add_be32(struct net_buf_simple *buf, u32_t val);
+
+/**
+ * @brief Push data to the beginning of the buffer.
+ *
+ * Modifies the data pointer and buffer length to account for more data
+ * in the beginning of the buffer.
+ *
+ * @param buf Buffer to update.
+ * @param len Number of bytes to add to the beginning.
+ *
+ * @return The new beginning of the buffer data.
+ */
+void *net_buf_simple_push(struct net_buf_simple *buf, size_t len);
+
+/**
+ * @brief Push 16-bit value to the beginning of the buffer
+ *
+ * Adds 16-bit value in little endian format to the beginning of the
+ * buffer.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be pushed to the buffer.
+ */
+void net_buf_simple_push_le16(struct net_buf_simple *buf, u16_t val);
+
+/**
+ * @brief Push 16-bit value to the beginning of the buffer
+ *
+ * Adds 16-bit value in big endian format to the beginning of the
+ * buffer.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be pushed to the buffer.
+ */
+void net_buf_simple_push_be16(struct net_buf_simple *buf, u16_t val);
+
+/**
+ * @brief Push 8-bit value to the beginning of the buffer
+ *
+ * Adds 8-bit value the beginning of the buffer.
+ *
+ * @param buf Buffer to update.
+ * @param val 8-bit value to be pushed to the buffer.
+ */
+void net_buf_simple_push_u8(struct net_buf_simple *buf, u8_t val);
+
+/**
+ * @brief Remove data from the beginning of the buffer.
+ *
+ * Removes data from the beginning of the buffer by modifying the data
+ * pointer and buffer length.
+ *
+ * @param buf Buffer to update.
+ * @param len Number of bytes to remove.
+ *
+ * @return New beginning of the buffer data.
+ */
+void *net_buf_simple_pull(struct net_buf_simple *buf, size_t len);
+
+/**
+ * @brief Remove a 8-bit value from the beginning of the buffer
+ *
+ * Same idea as with net_buf_simple_pull(), but a helper for operating
+ * on 8-bit values.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return The 8-bit removed value
+ */
+u8_t net_buf_simple_pull_u8(struct net_buf_simple *buf);
+
+/**
+ * @brief Remove and convert 16 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_simple_pull(), but a helper for operating
+ * on 16-bit little endian data.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return 16-bit value converted from little endian to host endian.
+ */
+u16_t net_buf_simple_pull_le16(struct net_buf_simple *buf);
+
+/**
+ * @brief Remove and convert 16 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_simple_pull(), but a helper for operating
+ * on 16-bit big endian data.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return 16-bit value converted from big endian to host endian.
+ */
+u16_t net_buf_simple_pull_be16(struct net_buf_simple *buf);
+
+/**
+ * @brief Remove and convert 32 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_simple_pull(), but a helper for operating
+ * on 32-bit little endian data.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return 32-bit value converted from little endian to host endian.
+ */
+u32_t net_buf_simple_pull_le32(struct net_buf_simple *buf);
+
+/**
+ * @brief Remove and convert 32 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_simple_pull(), but a helper for operating
+ * on 32-bit big endian data.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return 32-bit value converted from big endian to host endian.
+ */
+u32_t net_buf_simple_pull_be32(struct net_buf_simple *buf);
+
+/**
+ * @brief Get the tail pointer for a buffer.
+ *
+ * Get a pointer to the end of the data in a buffer.
+ *
+ * @param buf Buffer.
+ *
+ * @return Tail pointer for the buffer.
+ */
+static inline u8_t *net_buf_simple_tail(struct net_buf_simple *buf)
+{
+ return buf->data + buf->len;
+}
+
+/**
+ * @brief Check buffer headroom.
+ *
+ * Check how much free space there is in the beginning of the buffer.
+ *
+ * buf A valid pointer on a buffer
+ *
+ * @return Number of bytes available in the beginning of the buffer.
+ */
+size_t net_buf_simple_headroom(struct net_buf_simple *buf);
+
+/**
+ * @brief Check buffer tailroom.
+ *
+ * Check how much free space there is at the end of the buffer.
+ *
+ * @param buf A valid pointer on a buffer
+ *
+ * @return Number of bytes available at the end of the buffer.
+ */
+size_t net_buf_simple_tailroom(struct net_buf_simple *buf);
+
+/**
+ * @brief Parsing state of a buffer.
+ *
+ * This is used for temporarily storing the parsing state of a buffer
+ * while giving control of the parsing to a routine which we don't
+ * control.
+ */
+struct net_buf_simple_state {
+ /** Offset of the data pointer from the beginning of the storage */
+ u16_t offset;
+ /** Length of data */
+ u16_t len;
+};
+
+/**
+ * @brief Save the parsing state of a buffer.
+ *
+ * Saves the parsing state of a buffer so it can be restored later.
+ *
+ * @param buf Buffer from which the state should be saved.
+ * @param state Storage for the state.
+ */
+static inline void net_buf_simple_save(struct net_buf_simple *buf,
+ struct net_buf_simple_state *state)
+{
+ state->offset = net_buf_simple_headroom(buf);
+ state->len = buf->len;
+}
+
+/**
+ * @brief Restore the parsing state of a buffer.
+ *
+ * Restores the parsing state of a buffer from a state previously stored
+ * by net_buf_simple_save().
+ *
+ * @param buf Buffer to which the state should be restored.
+ * @param state Stored state.
+ */
+static inline void net_buf_simple_restore(struct net_buf_simple *buf,
+ struct net_buf_simple_state *state)
+{
+ buf->data = buf->__buf + state->offset;
+ buf->len = state->len;
+}
+
+/** Flag indicating that the buffer has associated fragments. Only used
+ * internally by the buffer handling code while the buffer is inside a
+ * FIFO, meaning this never needs to be explicitly set or unset by the
+ * net_buf API user. As long as the buffer is outside of a FIFO, i.e.
+ * in practice always for the user for this API, the buf->frags pointer
+ * should be used instead.
+ */
+#define NET_BUF_FRAGS BIT(0)
+
+/** @brief Network buffer representation.
+ *
+ * This struct is used to represent network buffers. Such buffers are
+ * normally defined through the NET_BUF_POOL_DEFINE() API and allocated
+ * using the net_buf_alloc() API.
+ */
+struct net_buf {
+ union {
+ /** Allow placing the buffer into sys_slist_t */
+ sys_snode_t node;
+
+ /** Fragments associated with this buffer. */
+ struct net_buf *frags;
+ };
+
+ /** Reference count. */
+ u8_t ref;
+
+ /* Change by Espressif. Net buf index */
+ u8_t index;
+
+ /** Bit-field of buffer flags. */
+ u8_t flags;
+
+ /** Where the buffer should go when freed up. */
+ struct net_buf_pool *pool_id;
+
+ /* Union for convenience access to the net_buf_simple members, also
+ * preserving the old API.
+ */
+ union {
+ /* The ABI of this struct must match net_buf_simple */
+ struct {
+ /** Pointer to the start of data in the buffer. */
+ u8_t *data;
+
+ /** Length of the data behind the data pointer. */
+ u16_t len;
+
+ /** Amount of data that this buffer can store. */
+ u16_t size;
+ };
+
+ struct net_buf_simple b;
+ };
+
+ /** Start of the data storage. Not to be accessed directly
+ * (the data pointer should be used instead).
+ */
+ u8_t __buf[0] __net_buf_align;
+
+ /** After __buf (as given by the "size" field, which can be 0),
+ * there may be so-called "user data", which is actually a system
+ * metadata for this buffer. This area can be accessed using
+ * net_buf_user_data(). (Its size is equal to
+ * this->pool->user_data_size.)
+ */
+};
+
+struct net_buf_pool {
+ /** LIFO to place the buffer into when free */
+ struct k_lifo free;
+
+ /** Number of buffers in pool */
+ const u16_t buf_count;
+
+ /** Number of uninitialized buffers */
+ u16_t uninit_count;
+
+ /** Data size of each buffer in the pool */
+ const u16_t buf_size;
+
+ /** Size of the user data associated with each buffer. */
+ const u16_t user_data_size;
+
+#if defined(CONFIG_NET_BUF_POOL_USAGE)
+ /** Amount of available buffers in the pool. */
+ s16_t avail_count;
+
+ /** Total size of the pool. */
+ const u16_t pool_size;
+
+ /** Name of the pool. Used when printing pool information. */
+ const char *name;
+#endif /* CONFIG_NET_BUF_POOL_USAGE */
+
+ /** Optional destroy callback when buffer is freed. */
+ void (*const destroy)(struct net_buf *buf);
+
+ /** Helper to access the start of storage (for net_buf_pool_init) */
+ struct net_buf *const __bufs;
+};
+
+#if defined(CONFIG_NET_BUF_POOL_USAGE)
+#define NET_BUF_POOL_INITIALIZER(_pool, _bufs, _count, _size, _ud_size, \
+ _destroy) \
+ { \
+ .free = _K_LIFO_INITIALIZER(_pool.free), \
+ .__bufs = (struct net_buf *)_bufs, \
+ .buf_count = _count, \
+ .uninit_count = _count, \
+ .avail_count = _count, \
+ .pool_size = sizeof(_net_buf_##_pool), \
+ .buf_size = _size, \
+ .user_data_size = _ud_size, \
+ .destroy = _destroy, \
+ .name = STRINGIFY(_pool), \
+ }
+#else
+#define NET_BUF_POOL_INITIALIZER(_pool, _bufs, _count, _size, _ud_size, \
+ _destroy) \
+ { \
+ .free = _K_LIFO_INITIALIZER(_pool.free), \
+ .__bufs = (struct net_buf *)_bufs, \
+ .buf_count = _count, \
+ .uninit_count = _count, \
+ .buf_size = _size, \
+ .user_data_size = _ud_size, \
+ .destroy = _destroy, \
+ }
+#endif /* CONFIG_NET_BUF_POOL_USAGE */
+
+/** @def NET_BUF_POOL_DEFINE
+ * @brief Define a new pool for buffers
+ *
+ * Defines a net_buf_pool struct and the necessary memory storage (array of
+ * structs) for the needed amount of buffers. After this,the buffers can be
+ * accessed from the pool through net_buf_alloc. The pool is defined as a
+ * static variable, so if it needs to be exported outside the current module
+ * this needs to happen with the help of a separate pointer rather than an
+ * extern declaration.
+ *
+ * If provided with a custom destroy callback this callback is
+ * responsible for eventually calling net_buf_destroy() to complete the
+ * process of returning the buffer to the pool.
+ *
+ * @param _name Name of the pool variable.
+ * @param _count Number of buffers in the pool.
+ * @param _size Maximum data size for each buffer.
+ * @param _ud_size Amount of user data space to reserve.
+ * @param _destroy Optional destroy callback when buffer is freed.
+ */
+/* ESP Toolchain doesn't support section */
+#define NET_BUF_POOL_DEFINE(_name, _count, _size, _ud_size, _destroy) \
+ static struct { \
+ struct net_buf buf; \
+ u8_t data[_size] __net_buf_align; \
+ u8_t ud[ROUND_UP(_ud_size, 4)] __net_buf_align; \
+ } _net_buf_##_name[_count]; \
+ struct net_buf_pool _name __net_buf_align \
+ __in_section(_net_buf_pool, static, _name) = \
+ NET_BUF_POOL_INITIALIZER(_name, _net_buf_##_name, \
+ _count, _size, _ud_size, _destroy)
+
+
+/**
+ * @brief Looks up a pool based on its ID.
+ *
+ * @param id Pool ID (e.g. from buf->pool_id).
+ *
+ * @return Pointer to pool.
+ */
+struct net_buf_pool *net_buf_pool_get(int id);
+
+/**
+ * @brief Get a zero-based index for a buffer.
+ *
+ * This function will translate a buffer into a zero-based index,
+ * based on its placement in its buffer pool. This can be useful if you
+ * want to associate an external array of meta-data contexts with the
+ * buffers of a pool.
+ *
+ * @param buf Network buffer.
+ *
+ * @return Zero-based index for the buffer.
+ */
+int net_buf_id(struct net_buf *buf);
+
+/**
+ * @brief Allocate a new buffer from a pool.
+ *
+ * Allocate a new buffer from a pool.
+ *
+ * @param pool Which pool to allocate the buffer from.
+ * @param timeout Affects the action taken should the pool be empty.
+ * If K_NO_WAIT, then return immediately. If K_FOREVER, then
+ * wait as long as necessary. Otherwise, wait up to the specified
+ * number of milliseconds before timing out.
+ *
+ * @return New buffer or NULL if out of buffers.
+ */
+#if defined(CONFIG_NET_BUF_LOG)
+struct net_buf *net_buf_alloc_debug(struct net_buf_pool *pool, s32_t timeout,
+ const char *func, int line);
+#define net_buf_alloc(_pool, _timeout) \
+ net_buf_alloc_debug(_pool, _timeout, __func__, __LINE__)
+#else
+struct net_buf *net_buf_alloc(struct net_buf_pool *pool, s32_t timeout);
+#endif
+
+/**
+ * @brief Get a buffer from a FIFO.
+ *
+ * Get buffer from a FIFO.
+ *
+ * @param fifo Which FIFO to take the buffer from.
+ * @param timeout Affects the action taken should the FIFO be empty.
+ * If K_NO_WAIT, then return immediately. If K_FOREVER, then wait as
+ * long as necessary. Otherwise, wait up to the specified number of
+ * milliseconds before timing out.
+ *
+ * @return New buffer or NULL if the FIFO is empty.
+ */
+#if defined(CONFIG_NET_BUF_LOG)
+struct net_buf *net_buf_get_debug(struct k_fifo *fifo, s32_t timeout,
+ const char *func, int line);
+#define net_buf_get(_fifo, _timeout) \
+ net_buf_get_debug(_fifo, _timeout, __func__, __LINE__)
+#else
+struct net_buf *net_buf_get(struct k_fifo *fifo, s32_t timeout);
+#endif
+
+/**
+ * @brief Destroy buffer from custom destroy callback
+ *
+ * This helper is only intended to be used from custom destroy callbacks.
+ * If no custom destroy callback is given to NET_BUF_POOL_DEFINE() then
+ * there is no need to use this API.
+ *
+ * @param buf Buffer to destroy.
+ */
+static inline void net_buf_destroy(struct net_buf *buf)
+{
+ //struct net_buf_pool *pool = net_buf_pool_get(buf->pool_id);
+ // TODO: should replace the following funcion in our system
+ //k_lifo_put(&pool->free, buf);
+}
+
+/**
+ * @brief Reset buffer
+ *
+ * Reset buffer data and flags so it can be reused for other purposes.
+ *
+ * @param buf Buffer to reset.
+ */
+void net_buf_reset(struct net_buf *buf);
+
+/**
+ * @brief Initialize buffer with the given headroom.
+ *
+ * Initializes a buffer with a given headroom. The buffer is not expected to
+ * contain any data when this API is called.
+ *
+ * @param buf Buffer to initialize.
+ * @param reserve How much headroom to reserve.
+ */
+void net_buf_reserve(struct net_buf *buf, size_t reserve);
+
+/**
+ * @brief Put a buffer into a list
+ *
+ * Put a buffer to the end of a list. If the buffer contains follow-up
+ * fragments this function will take care of inserting them as well
+ * into the list.
+ *
+ * @param list Which list to append the buffer to.
+ * @param buf Buffer.
+ */
+void net_buf_slist_put(sys_slist_t *list, struct net_buf *buf);
+
+/**
+ * @brief Get a buffer from a list.
+ *
+ * Get buffer from a list. If the buffer had any fragments, these will
+ * automatically be recovered from the list as well and be placed to
+ * the buffer's fragment list.
+ *
+ * @param list Which list to take the buffer from.
+ *
+ * @return New buffer or NULL if the FIFO is empty.
+ */
+struct net_buf *net_buf_slist_get(sys_slist_t *list);
+
+/**
+ * @brief Put a buffer into a FIFO
+ *
+ * Put a buffer to the end of a FIFO. If the buffer contains follow-up
+ * fragments this function will take care of inserting them as well
+ * into the FIFO.
+ *
+ * @param fifo Which FIFO to put the buffer to.
+ * @param buf Buffer.
+ */
+void net_buf_put(struct k_fifo *fifo, struct net_buf *buf);
+
+/**
+ * @brief Decrements the reference count of a buffer.
+ *
+ * Decrements the reference count of a buffer and puts it back into the
+ * pool if the count reaches zero.
+ *
+ * @param buf A valid pointer on a buffer
+ */
+#if defined(CONFIG_NET_BUF_LOG)
+void net_buf_unref_debug(struct net_buf *buf, const char *func, int line);
+#define net_buf_unref(_buf) \
+ net_buf_unref_debug(_buf, __func__, __LINE__)
+#else
+void net_buf_unref(struct net_buf *buf);
+#endif
+
+/**
+ * @brief Increment the reference count of a buffer.
+ *
+ * @param buf A valid pointer on a buffer
+ *
+ * @return the buffer newly referenced
+ */
+struct net_buf *net_buf_ref(struct net_buf *buf);
+
+/**
+ * @brief Duplicate buffer
+ *
+ * Duplicate given buffer including any data and headers currently stored.
+ *
+ * @param buf A valid pointer on a buffer
+ * @param timeout Affects the action taken should the pool be empty.
+ * If K_NO_WAIT, then return immediately. If K_FOREVER, then
+ * wait as long as necessary. Otherwise, wait up to the specified
+ * number of milliseconds before timing out.
+ *
+ * @return Duplicated buffer or NULL if out of buffers.
+ */
+struct net_buf *net_buf_clone(struct net_buf *buf, s32_t timeout);
+
+/**
+ * @brief Get a pointer to the user data of a buffer.
+ *
+ * @param buf A valid pointer on a buffer
+ *
+ * @return Pointer to the user data of the buffer.
+ */
+static inline void *net_buf_user_data(struct net_buf *buf)
+{
+ return (void *)ROUND_UP((buf->__buf + buf->size), sizeof(int));
+}
+
+/**
+ * @def net_buf_add
+ * @brief Prepare data to be added at the end of the buffer
+ *
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param len Number of bytes to increment the length with.
+ *
+ * @return The original tail of the buffer.
+ */
+#define net_buf_add(buf, len) net_buf_simple_add(&(buf)->b, len)
+
+/**
+ * @def net_buf_add_mem
+ * @brief Copy bytes from memory to the end of the buffer
+ *
+ * Copies the given number of bytes to the end of the buffer. Increments the
+ * data length of the buffer to account for more data at the end.
+ *
+ * @param buf Buffer to update.
+ * @param mem Location of data to be added.
+ * @param len Length of data to be added
+ *
+ * @return The original tail of the buffer.
+ */
+#define net_buf_add_mem(buf, mem, len) net_buf_simple_add_mem(&(buf)->b, \
+ mem, len)
+
+/**
+ * @def net_buf_add_u8
+ * @brief Add (8-bit) byte at the end of the buffer
+ *
+ * Adds a byte at the end of the buffer. Increments the data length of
+ * the buffer to account for more data at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val byte value to be added.
+ *
+ * @return Pointer to the value added
+ */
+#define net_buf_add_u8(buf, val) net_buf_simple_add_u8(&(buf)->b, val)
+
+/**
+ * @def net_buf_add_le16
+ * @brief Add 16-bit value at the end of the buffer
+ *
+ * Adds 16-bit value in little endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be added.
+ */
+#define net_buf_add_le16(buf, val) net_buf_simple_add_le16(&(buf)->b, val)
+
+/**
+ * @def net_buf_add_be16
+ * @brief Add 16-bit value at the end of the buffer
+ *
+ * Adds 16-bit value in big endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be added.
+ */
+#define net_buf_add_be16(buf, val) net_buf_simple_add_be16(&(buf)->b, val)
+
+/**
+ * @def net_buf_add_le32
+ * @brief Add 32-bit value at the end of the buffer
+ *
+ * Adds 32-bit value in little endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 32-bit value to be added.
+ */
+#define net_buf_add_le32(buf, val) net_buf_simple_add_le32(&(buf)->b, val)
+
+/**
+ * @def net_buf_add_be32
+ * @brief Add 32-bit value at the end of the buffer
+ *
+ * Adds 32-bit value in big endian format at the end of buffer.
+ * Increments the data length of a buffer to account for more data
+ * at the end.
+ *
+ * @param buf Buffer to update.
+ * @param val 32-bit value to be added.
+ */
+#define net_buf_add_be32(buf, val) net_buf_simple_add_be32(&(buf)->b, val)
+
+/**
+ * @def net_buf_push
+ * @brief Push data to the beginning of the buffer.
+ *
+ * Modifies the data pointer and buffer length to account for more data
+ * in the beginning of the buffer.
+ *
+ * @param buf Buffer to update.
+ * @param len Number of bytes to add to the beginning.
+ *
+ * @return The new beginning of the buffer data.
+ */
+#define net_buf_push(buf, len) net_buf_simple_push(&(buf)->b, len)
+
+/**
+ * @def net_buf_push_le16
+ * @brief Push 16-bit value to the beginning of the buffer
+ *
+ * Adds 16-bit value in little endian format to the beginning of the
+ * buffer.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be pushed to the buffer.
+ */
+#define net_buf_push_le16(buf, val) net_buf_simple_push_le16(&(buf)->b, val)
+
+/**
+ * @def net_buf_push_be16
+ * @brief Push 16-bit value to the beginning of the buffer
+ *
+ * Adds 16-bit value in little endian format to the beginning of the
+ * buffer.
+ *
+ * @param buf Buffer to update.
+ * @param val 16-bit value to be pushed to the buffer.
+ */
+#define net_buf_push_be16(buf, val) net_buf_simple_push_be16(&(buf)->b, val)
+
+/**
+ * @def net_buf_push_u8
+ * @brief Push 8-bit value to the beginning of the buffer
+ *
+ * Adds 8-bit value the beginning of the buffer.
+ *
+ * @param buf Buffer to update.
+ * @param val 8-bit value to be pushed to the buffer.
+ */
+#define net_buf_push_u8(buf, val) net_buf_simple_push_u8(&(buf)->b, val)
+
+/**
+ * @def net_buf_pull
+ * @brief Remove data from the beginning of the buffer.
+ *
+ * Removes data from the beginning of the buffer by modifying the data
+ * pointer and buffer length.
+ *
+ * @param buf Buffer to update.
+ * @param len Number of bytes to remove.
+ *
+ * @return New beginning of the buffer data.
+ */
+#define net_buf_pull(buf, len) net_buf_simple_pull(&(buf)->b, len)
+
+/**
+ * @def net_buf_pull_u8
+ * @brief Remove a 8-bit value from the beginning of the buffer
+ *
+ * Same idea as with net_buf_pull(), but a helper for operating on
+ * 8-bit values.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return The 8-bit removed value
+ */
+#define net_buf_pull_u8(buf) net_buf_simple_pull_u8(&(buf)->b)
+
+/**
+ * @def net_buf_pull_le16
+ * @brief Remove and convert 16 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_pull(), but a helper for operating on
+ * 16-bit little endian data.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return 16-bit value converted from little endian to host endian.
+ */
+#define net_buf_pull_le16(buf) net_buf_simple_pull_le16(&(buf)->b)
+
+/**
+ * @def net_buf_pull_be16
+ * @brief Remove and convert 16 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_pull(), but a helper for operating on
+ * 16-bit big endian data.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return 16-bit value converted from big endian to host endian.
+ */
+#define net_buf_pull_be16(buf) net_buf_simple_pull_be16(&(buf)->b)
+
+/**
+ * @def net_buf_pull_le32
+ * @brief Remove and convert 32 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_pull(), but a helper for operating on
+ * 32-bit little endian data.
+ *
+ * @param buf A valid pointer on a buffer.
+ *
+ * @return 32-bit value converted from little endian to host endian.
+ */
+#define net_buf_pull_le32(buf) net_buf_simple_pull_le32(&(buf)->b)
+
+/**
+ * @def net_buf_pull_be32
+ * @brief Remove and convert 32 bits from the beginning of the buffer.
+ *
+ * Same idea as with net_buf_pull(), but a helper for operating on
+ * 32-bit big endian data.
+ *
+ * @param buf A valid pointer on a buffer
+ *
+ * @return 32-bit value converted from big endian to host endian.
+ */
+#define net_buf_pull_be32(buf) net_buf_simple_pull_be32(&(buf)->b)
+
+/**
+ * @def net_buf_tailroom
+ * @brief Check buffer tailroom.
+ *
+ * Check how much free space there is at the end of the buffer.
+ *
+ * @param buf A valid pointer on a buffer
+ *
+ * @return Number of bytes available at the end of the buffer.
+ */
+#define net_buf_tailroom(buf) net_buf_simple_tailroom(&(buf)->b)
+
+/**
+ * @def net_buf_headroom
+ * @brief Check buffer headroom.
+ *
+ * Check how much free space there is in the beginning of the buffer.
+ *
+ * buf A valid pointer on a buffer
+ *
+ * @return Number of bytes available in the beginning of the buffer.
+ */
+#define net_buf_headroom(buf) net_buf_simple_headroom(&(buf)->b)
+
+/**
+ * @def net_buf_tail
+ * @brief Get the tail pointer for a buffer.
+ *
+ * Get a pointer to the end of the data in a buffer.
+ *
+ * @param buf Buffer.
+ *
+ * @return Tail pointer for the buffer.
+ */
+#define net_buf_tail(buf) net_buf_simple_tail(&(buf)->b)
+
+/** @brief Find the last fragment in the fragment list.
+ *
+ * @return Pointer to last fragment in the list.
+ */
+struct net_buf *net_buf_frag_last(struct net_buf *frags);
+
+/** @brief Insert a new fragment to a chain of bufs.
+ *
+ * Insert a new fragment into the buffer fragments list after the parent.
+ *
+ * Note: This function takes ownership of the fragment reference so the
+ * caller is not required to unref.
+ *
+ * @param parent Parent buffer/fragment.
+ * @param frag Fragment to insert.
+ */
+void net_buf_frag_insert(struct net_buf *parent, struct net_buf *frag);
+
+/** @brief Add a new fragment to the end of a chain of bufs.
+ *
+ * Append a new fragment into the buffer fragments list.
+ *
+ * Note: This function takes ownership of the fragment reference so the
+ * caller is not required to unref.
+ *
+ * @param head Head of the fragment chain.
+ * @param frag Fragment to add.
+ *
+ * @return New head of the fragment chain. Either head (if head
+ * was non-NULL) or frag (if head was NULL).
+ */
+struct net_buf *net_buf_frag_add(struct net_buf *head, struct net_buf *frag);
+
+/** @brief Delete existing fragment from a chain of bufs.
+ *
+ * @param parent Parent buffer/fragment, or NULL if there is no parent.
+ * @param frag Fragment to delete.
+ *
+ * @return Pointer to the buffer following the fragment, or NULL if it
+ * had no further fragments.
+ */
+#if defined(CONFIG_NET_BUF_LOG)
+struct net_buf *net_buf_frag_del_debug(struct net_buf *parent,
+ struct net_buf *frag,
+ const char *func, int line);
+#define net_buf_frag_del(_parent, _frag) \
+ net_buf_frag_del_debug(_parent, _frag, __func__, __LINE__)
+#else
+struct net_buf *net_buf_frag_del(struct net_buf *parent, struct net_buf *frag);
+#endif
+
+/** @brief Calculate amount of bytes stored in fragments.
+ *
+ * Calculates the total amount of data stored in the given buffer and the
+ * fragments linked to it.
+ *
+ * @param buf Buffer to start off with.
+ *
+ * @return Number of bytes in the buffer and its fragments.
+ */
+static inline size_t net_buf_frags_len(struct net_buf *buf)
+{
+ size_t bytes = 0;
+
+ while (buf) {
+ bytes += buf->len;
+ buf = buf->frags;
+ }
+
+ return bytes;
+}
+
+/**
+ * @}
+ */
+
+#ifdef __cplusplus
+}
+#endif
+
+
+
+
+#endif /* __NET_BUF_H */
+
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_hci.h b/components/bt/ble_mesh/mesh_core/include/mesh_hci.h
new file mode 100644
index 0000000000..7e16eef3ff
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_hci.h
@@ -0,0 +1,239 @@
+/*
+ * Copyright (c) 2015-2016 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __MESH_HCI_H
+#define __MESH_HCI_H
+
+#include "mesh_kernel.h"
+#include "mesh_bearer_adapt.h"
+#include "mesh_atomic.h"
+
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+/* Porting form zephyr/subsys/bluetooth/host/hci_core.h */
+
+#if defined(CONFIG_BT_BREDR)
+#define LMP_FEAT_PAGES_COUNT 3
+#else
+#define LMP_FEAT_PAGES_COUNT 1
+#endif
+
+/* bt_dev flags: the flags defined here represent BT controller state */
+enum {
+ BT_DEV_ENABLE,
+ BT_DEV_READY,
+ BT_DEV_ID_STATIC_RANDOM,
+ BT_DEV_HAS_PUB_KEY,
+ BT_DEV_PUB_KEY_BUSY,
+
+ BT_DEV_ADVERTISING,
+ BT_DEV_KEEP_ADVERTISING,
+ BT_DEV_SCANNING,
+ BT_DEV_EXPLICIT_SCAN,
+ BT_DEV_ACTIVE_SCAN,
+ BT_DEV_SCAN_FILTER_DUP,
+
+ BT_DEV_RPA_VALID,
+
+ BT_DEV_ID_PENDING,
+
+#if defined(CONFIG_BT_BREDR)
+ BT_DEV_ISCAN,
+ BT_DEV_PSCAN,
+ BT_DEV_INQUIRY,
+#endif /* CONFIG_BT_BREDR */
+
+ /* Total number of flags - must be at the end of the enum */
+ BT_DEV_NUM_FLAGS,
+};
+
+struct bt_dev_le {
+ /* LE features */
+ u8_t features[8];
+ /* LE states */
+ u64_t states;
+
+#if defined(CONFIG_BT_CONN)
+ /* Controller buffer information */
+ u16_t mtu;
+ struct k_sem pkts;
+#endif /* CONFIG_BT_CONN */
+
+#if defined(CONFIG_BT_SMP)
+ /* Size of the the controller resolving list */
+ u8_t rl_size;
+ /* Number of entries in the resolving list. rl_entries > rl_size
+ * means that host-side resolving is used.
+ */
+ u8_t rl_entries;
+#endif /* CONFIG_BT_SMP */
+};
+
+#if defined(CONFIG_BT_BREDR)
+struct bt_dev_br {
+ /* Max controller's acceptable ACL packet length */
+ u16_t mtu;
+ struct k_sem pkts;
+ u16_t esco_pkt_type;
+};
+#endif
+
+/* The theoretical max for these is 8 and 64, but there's no point
+ * in allocating the full memory if we only support a small subset.
+ * These values must be updated whenever the host implementation is
+ * extended beyond the current values.
+ */
+#define BT_DEV_VS_FEAT_MAX 1
+#define BT_DEV_VS_CMDS_MAX 2
+
+/* State tracking for the local Bluetooth controller */
+struct bt_dev {
+ /* Local Identity Address */
+ bt_addr_le_t id_addr;
+
+ /* Current local Random Address */
+ bt_addr_le_t random_addr;
+
+ /* Controller version & manufacturer information */
+ u8_t hci_version;
+ u8_t lmp_version;
+ u16_t hci_revision;
+ u16_t lmp_subversion;
+ u16_t manufacturer;
+
+ /* LMP features (pages 0, 1, 2) */
+ u8_t features[LMP_FEAT_PAGES_COUNT][8];
+
+ /* Supported commands */
+ u8_t supported_commands[64];
+
+#if defined(CONFIG_BT_HCI_VS_EXT)
+ /* Vendor HCI support */
+ u8_t vs_features[BT_DEV_VS_FEAT_MAX];
+ u8_t vs_commands[BT_DEV_VS_CMDS_MAX];
+#endif
+
+#if 0
+ struct k_work init;
+#endif /* if 0 */
+ ATOMIC_DEFINE(flags, BT_DEV_NUM_FLAGS);
+
+ /* LE controller specific features */
+ struct bt_dev_le le;
+
+#if defined(CONFIG_BT_BREDR)
+ /* BR/EDR controller specific features */
+ struct bt_dev_br br;
+#endif
+
+#if 0
+ /* Number of commands controller can accept */
+ struct k_sem ncmd_sem;
+
+ /* Last sent HCI command */
+ struct net_buf *sent_cmd;
+#endif /* if 0 */
+
+#if !defined(CONFIG_BT_RECV_IS_RX_THREAD)
+ /* Queue for incoming HCI events & ACL data */
+ struct k_fifo rx_queue;
+#endif
+
+#if 0
+
+ /* Queue for high priority HCI events which may unlock waiters
+ * in other threads. Such events include Number of Completed
+ * Packets, as well as the Command Complete/Status events.
+ */
+ struct k_fifo rx_prio_queue;
+
+ /* Queue for outgoing HCI commands */
+ struct k_fifo cmd_tx_queue;
+
+ /* Registered HCI driver */
+ const struct bt_hci_driver *drv;
+#endif /* #if 0 */
+#if defined(CONFIG_BT_PRIVACY)
+ /* Local Identity Resolving Key */
+ u8_t irk[16];
+
+ /* Work used for RPA rotation */
+ struct k_delayed_work rpa_update;
+#endif
+};
+
+/* Added by Espressif */
+extern struct bt_dev bt_dev;
+
+void mesh_hci_init(void);
+
+
+/*Porting from zephyr/subsys/bluetooth/host/hci_core.h */
+/* HCI version from Assigned Numbers */
+#define BT_HCI_VERSION_1_0B 0
+#define BT_HCI_VERSION_1_1 1
+#define BT_HCI_VERSION_1_2 2
+#define BT_HCI_VERSION_2_0 3
+#define BT_HCI_VERSION_2_1 4
+#define BT_HCI_VERSION_3_0 5
+#define BT_HCI_VERSION_4_0 6
+#define BT_HCI_VERSION_4_1 7
+#define BT_HCI_VERSION_4_2 8
+#define BT_HCI_VERSION_5_0 9
+
+/* OpCode Group Fields */
+#define BT_OGF_LINK_CTRL 0x01
+#define BT_OGF_BASEBAND 0x03
+#define BT_OGF_INFO 0x04
+#define BT_OGF_STATUS 0x05
+#define BT_OGF_LE 0x08
+#define BT_OGF_VS 0x3f
+
+/* Construct OpCode from OGF and OCF */
+#define BT_OP(ogf, ocf) ((ocf) | ((ogf) << 10))
+
+/* Obtain OGF from OpCode */
+#define BT_OGF(opcode) (((opcode) >> 10) & BIT_MASK(6))
+/* Obtain OCF from OpCode */
+#define BT_OCF(opcode) ((opcode) & BIT_MASK(10))
+
+#define BT_HCI_OP_LE_SET_ADV_PARAM BT_OP(BT_OGF_LE, 0x0006)
+struct bt_hci_cp_le_set_adv_param {
+ u16_t min_interval;
+ u16_t max_interval;
+ u8_t type;
+ u8_t own_addr_type;
+ bt_addr_le_t direct_addr;
+ u8_t channel_map;
+ u8_t filter_policy;
+} __packed;
+
+#define BT_HCI_OP_LE_SET_ADV_DATA BT_OP(BT_OGF_LE, 0x0008)
+struct bt_hci_cp_le_set_adv_data {
+ u8_t len;
+ u8_t data[31];
+} __packed;
+
+#define BT_HCI_OP_LE_SET_SCAN_RSP_DATA BT_OP(BT_OGF_LE, 0x0009)
+struct bt_hci_cp_le_set_scan_rsp_data {
+ u8_t len;
+ u8_t data[31];
+} __packed;
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __MESH_HCI_H */
+
+
+
+
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_kernel.h b/components/bt/ble_mesh/mesh_core/include/mesh_kernel.h
new file mode 100644
index 0000000000..efa8d79b5f
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_kernel.h
@@ -0,0 +1,472 @@
+/*
+ * Copyright (c) 2016, Wind River Systems, Inc.
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef _MESH_KERNEL_
+#define _MESH_KERNEL_
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "../../../bluedroid/osi/include/osi/mutex.h"
+#include "mesh_types.h"
+#include "mesh_slist.h"
+#include "dlist.h"
+#include "mesh_atomic.h"
+
+typedef int atomic_t;
+typedef atomic_t atomic_val_t;
+
+/* number of nsec per usec */
+#define NSEC_PER_USEC 1000
+
+/* number of microseconds per millisecond */
+#define USEC_PER_MSEC 1000
+
+/* number of milliseconds per second */
+#define MSEC_PER_SEC 1000
+
+/* number of microseconds per second */
+#define USEC_PER_SEC ((USEC_PER_MSEC) * (MSEC_PER_SEC))
+
+/* number of nanoseconds per second */
+#define NSEC_PER_SEC ((NSEC_PER_USEC) * (USEC_PER_MSEC) * (MSEC_PER_SEC))
+
+#define ATOMIC_MASK(bit) (1 << ((bit) & (ATOMIC_BITS - 1)))
+#define ATOMIC_ELEM(addr, bit) ((addr) + ((bit) / ATOMIC_BITS))
+
+/* timeout is not in use */
+#define _INACTIVE (-1)
+
+/**
+ * @} end defgroup stack_apis
+ */
+struct k_work;
+
+/**
+ * @typedef k_work_handler_t
+ * @brief Work item handler function type.
+ *
+ * A work item's handler function is executed by a workqueue's thread
+ * when the work item is processed by the workqueue.
+ *
+ * @param work Address of the work item.
+ *
+ * @return N/A
+ */
+typedef void (*k_work_handler_t)(struct k_work *work);
+
+
+typedef sys_dlist_t _wait_q_t;
+
+struct k_work {
+ void *_reserved; /* Used by k_queue implementation. */
+ k_work_handler_t handler;
+ atomic_t flags[1];
+ int index;
+};
+
+#define _K_WORK_INITIALIZER(work_handler) \
+ { \
+ ._reserved = NULL, \
+ .handler = work_handler, \
+ .flags = { 0 } \
+ }
+
+struct _timeout;
+
+typedef void (*_timeout_func_t)(struct _timeout *t);
+
+struct _timeout {
+ sys_dnode_t node;
+ //struct k_thread *thread;
+ sys_dlist_t *wait_q;
+ s32_t delta_ticks_from_prev;
+ _timeout_func_t func;
+};
+
+/**
+ * @brief Define an array of atomic variables.
+ *
+ * This macro defines an array of atomic variables containing at least
+ * @a num_bits bits.
+ *
+ * @note
+ * If used from file scope, the bits of the array are initialized to zero;
+ * if used from within a function, the bits are left uninitialized.
+ *
+ * @param name Name of array of atomic variables.
+ * @param num_bits Number of bits needed.
+ */
+#define ATOMIC_DEFINE(name, num_bits) \
+ atomic_t name[1 + ((num_bits) - 1) / ATOMIC_BITS]
+
+/**
+ * @brief Generate null timeout delay.
+ *
+ * This macro generates a timeout delay that that instructs a kernel API
+ * not to wait if the requested operation cannot be performed immediately.
+ *
+ * @return Timeout delay value.
+ */
+#define K_NO_WAIT 0
+
+/**
+ * @brief Generate timeout delay from milliseconds.
+ *
+ * This macro generates a timeout delay that that instructs a kernel API
+ * to wait up to @a ms milliseconds to perform the requested operation.
+ *
+ * @param ms Duration in milliseconds.
+ *
+ * @return Timeout delay value.
+ */
+#define K_MSEC(ms) (ms)
+
+/**
+ * @brief Generate timeout delay from seconds.
+ *
+ * This macro generates a timeout delay that that instructs a kernel API
+ * to wait up to @a s seconds to perform the requested operation.
+ *
+ * @param s Duration in seconds.
+ *
+ * @return Timeout delay value.
+ */
+#define K_SECONDS(s) K_MSEC((s) * MSEC_PER_SEC)
+
+/**
+ * @brief Generate timeout delay from minutes.
+ *
+ * This macro generates a timeout delay that that instructs a kernel API
+ * to wait up to @a m minutes to perform the requested operation.
+ *
+ * @param m Duration in minutes.
+ *
+ * @return Timeout delay value.
+ */
+#define K_MINUTES(m) K_SECONDS((m) * 60)
+
+/**
+ * @brief Generate timeout delay from hours.
+ *
+ * This macro generates a timeout delay that that instructs a kernel API
+ * to wait up to @a h hours to perform the requested operation.
+ *
+ * @param h Duration in hours.
+ *
+ * @return Timeout delay value.
+ */
+#define K_HOURS(h) K_MINUTES((h) * 60)
+
+/**
+ * @brief Generate infinite timeout delay.
+ *
+ * This macro generates a timeout delay that that instructs a kernel API
+ * to wait as long as necessary to perform the requested operation.
+ *
+ * @return Timeout delay value.
+ */
+#define K_FOREVER (-1)
+
+
+/**
+ * @brief Get system uptime (32-bit version).
+ *
+ * This routine returns the lower 32-bits of the elapsed time since the system
+ * booted, in milliseconds.
+ *
+ * This routine can be more efficient than k_uptime_get(), as it reduces the
+ * need for interrupt locking and 64-bit math. However, the 32-bit result
+ * cannot hold a system uptime time larger than approximately 50 days, so the
+ * caller must handle possible rollovers.
+ *
+ * @return Current uptime.
+ */
+u32_t k_uptime_get_32(void);
+
+struct k_sem {
+ osi_mutex_t mutex;
+};
+
+struct k_queue {
+ sys_slist_t data_q;
+ union {
+ _wait_q_t wait_q;
+ // TODO: The following not used now, delete it temporarily.
+ //_POLL_EVENT;
+ };
+ // TODO: The following not used now, delete it temporarily.
+ //_OBJECT_TRACING_NEXT_PTR(k_queue);
+};
+
+struct k_work_q {
+ struct k_queue queue;
+};
+
+
+
+/**
+ * @cond INTERNAL_HIDDEN
+ */
+
+struct k_fifo {
+ struct k_queue _queue;
+};
+
+struct k_lifo {
+ struct k_queue _queue;
+};
+
+struct k_delayed_work {
+ struct k_work work;
+ struct _timeout timeout;
+ struct k_work_q *work_q;
+};
+
+#define _K_QUEUE_INITIALIZER(obj) \
+ { \
+ .wait_q = SYS_DLIST_STATIC_INIT(&obj.wait_q), \
+ .data_q = SYS_SLIST_STATIC_INIT(&obj.data_q), \
+ }
+
+
+#define _K_FIFO_INITIALIZER(obj) \
+ { \
+ ._queue = _K_QUEUE_INITIALIZER(obj._queue) \
+ }
+
+#define _K_LIFO_INITIALIZER(obj) \
+ { \
+ ._queue = _K_QUEUE_INITIALIZER(obj._queue) \
+ }
+
+
+/**
+ * @brief Statically define and initialize a fifo.
+ *
+ * The fifo can be accessed outside the module where it is defined using:
+ *
+ * @code extern struct k_fifo ; @endcode
+ *
+ * @param name Name of the fifo.
+ */
+#define K_FIFO_DEFINE(name) \
+ struct k_fifo name \
+ __in_section(_k_queue, static, name) = \
+ _K_FIFO_INITIALIZER(name)
+
+
+/* initialize the timeouts part of k_thread when enabled in the kernel */
+
+static inline void _init_timeout(struct _timeout *t, _timeout_func_t func)
+{
+ /*
+ * Must be initialized here and when dequeueing a timeout so that code
+ * not dealing with timeouts does not have to handle this, such as when
+ * waiting forever on a semaphore.
+ */
+ t->delta_ticks_from_prev = _INACTIVE;
+
+ /*
+ * Must be initialized here so that k_wakeup can
+ * verify the thread is not on a wait queue before aborting a timeout.
+ */
+ t->wait_q = NULL;
+
+ /*
+ * Function must be initialized before being potentially called.
+ */
+ t->func = func;
+
+ /*
+ * These are initialized when enqueing on the timeout queue:
+ *
+ * thread->timeout.node.next
+ * thread->timeout.node.prev
+ */
+}
+
+unsigned int irq_lock(void);
+
+void irq_unlock(unsigned int key);
+
+void mesh_k_init(void);
+
+/**
+ * @brief Submit a delayed work item to the system workqueue.
+ *
+ * This routine schedules work item @a work to be processed by the system
+ * workqueue after a delay of @a delay milliseconds. The routine initiates
+ * an asynchronous countdown for the work item and then returns to the caller.
+ * Only when the countdown completes is the work item actually submitted to
+ * the workqueue and becomes pending.
+ *
+ * Submitting a previously submitted delayed work item that is still
+ * counting down cancels the existing submission and restarts the countdown
+ * using the new delay. If the work item is currently pending on the
+ * workqueue's queue because the countdown has completed it is too late to
+ * resubmit the item, and resubmission fails without impacting the work item.
+ * If the work item has already been processed, or is currently being processed,
+ * its work is considered complete and the work item can be resubmitted.
+ *
+ * @warning
+ * Work items submitted to the system workqueue should avoid using handlers
+ * that block or yield since this may prevent the system workqueue from
+ * processing other work items in a timely manner.
+ *
+ * @note Can be called by ISRs.
+ *
+ * @param work Address of delayed work item.
+ * @param delay Delay before submitting the work item (in milliseconds).
+ *
+ * @retval 0 Work item countdown started.
+ * @retval -EINPROGRESS Work item is already pending.
+ * @retval -EINVAL Work item is being processed or has completed its work.
+ * @retval -EADDRINUSE Work item is pending on a different workqueue.
+ */
+int k_delayed_work_submit(struct k_delayed_work *work,
+ s32_t delay);
+
+/**
+ * @brief Get time remaining before a delayed work gets scheduled.
+ *
+ * This routine computes the (approximate) time remaining before a
+ * delayed work gets executed. If the delayed work is not waiting to be
+ * scheduled, it returns zero.
+ *
+ * @param work Delayed work item.
+ *
+ * @return Remaining time (in milliseconds).
+ */
+s32_t k_delayed_work_remaining_get(struct k_delayed_work *work);
+
+/**
+ * @brief Submit a work item to the system workqueue.
+ *
+ * This routine submits work item @a work to be processed by the system
+ * workqueue. If the work item is already pending in the workqueue's queue
+ * as a result of an earlier submission, this routine has no effect on the
+ * work item. If the work item has already been processed, or is currently
+ * being processed, its work is considered complete and the work item can be
+ * resubmitted.
+ *
+ * @warning
+ * Work items submitted to the system workqueue should avoid using handlers
+ * that block or yield since this may prevent the system workqueue from
+ * processing other work items in a timely manner.
+ *
+ * @note Can be called by ISRs.
+ *
+ * @param work Address of work item.
+ *
+ * @return N/A
+ */
+static inline void k_work_submit(struct k_work *work)
+{
+ // TODO:
+ if (work != NULL) {
+ work->handler(work);
+ }
+
+ return;
+ //k_work_submit_to_queue(&k_sys_work_q, work);
+}
+
+/**
+ * @brief Initialize a work item.
+ *
+ * This routine initializes a workqueue work item, prior to its first use.
+ *
+ * @param work Address of work item.
+ * @param handler Function to invoke each time work item is processed.
+ *
+ * @return N/A
+ */
+static inline void k_work_init(struct k_work *work, k_work_handler_t handler)
+{
+ work->handler = handler;
+}
+
+int k_delayed_work_cancel(struct k_delayed_work *work);
+
+int k_delayed_work_free(struct k_delayed_work *work);
+
+void k_delayed_work_init(struct k_delayed_work *work, k_work_handler_t handler);
+
+/**
+ * @brief Get system uptime.
+ *
+ * This routine returns the elapsed time since the system booted,
+ * in milliseconds.
+ *
+ * @return Current uptime.
+ */
+s64_t k_uptime_get(void);
+
+/**
+ * @brief Put the current thread to sleep.
+ *
+ * This routine puts the current thread to sleep for @a duration
+ * milliseconds.
+ *
+ * @param duration Number of milliseconds to sleep.
+ *
+ * @return N/A
+ */
+void k_sleep(s32_t duration);
+
+/**
+ * @brief Give a semaphore.
+ *
+ * This routine gives @a sem, unless the semaphore is already at its maximum
+ * permitted count.
+ *
+ * @note Can be called by ISRs.
+ *
+ * @param sem Address of the semaphore.
+ *
+ * @return N/A
+ */
+void k_sem_give(struct k_sem *sem);
+
+/**
+ * @brief Initialize a semaphore.
+ *
+ * This routine initializes a semaphore object, prior to its first use.
+ *
+ * @param sem Address of the semaphore.
+ * @param initial_count Initial semaphore count.
+ * @param limit Maximum permitted semaphore count.
+ *
+ * @return N/A
+ */
+void k_sem_init(struct k_sem *sem, unsigned int initial_count,
+ unsigned int limit);
+
+/**
+ * @brief Take a semaphore.
+ *
+ * This routine takes @a sem.
+ *
+ * @note Can be called by ISRs, but @a timeout must be set to K_NO_WAIT.
+ *
+ * @param sem Address of the semaphore.
+ * @param timeout Waiting period to take the semaphore (in milliseconds),
+ * or one of the special values K_NO_WAIT and K_FOREVER.
+ *
+ * @note When porting code from the nanokernel legacy API to the new API, be
+ * careful with the return value of this function. The return value is the
+ * reverse of the one of nano_sem_take family of APIs: 0 means success, and
+ * non-zero means failure, while the nano_sem_take family returns 1 for success
+ * and 0 for failure.
+ *
+ * @retval 0 Semaphore taken.
+ * @retval -EBUSY Returned without waiting.
+ * @retval -EAGAIN Waiting period timed out.
+ */
+int k_sem_take(struct k_sem *sem, s32_t timeout);
+
+#endif /* #ifndef _MESH_KERNEL_ */
+
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_main.h b/components/bt/ble_mesh/mesh_core/include/mesh_main.h
new file mode 100644
index 0000000000..ae84da6b11
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_main.h
@@ -0,0 +1,534 @@
+/** @file
+ * @brief Bluetooth Mesh Profile APIs.
+ */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __BT_MESH_MAIN_H
+#define __BT_MESH_MAIN_H
+
+#include "mesh_util.h"
+#include "mesh_access.h"
+#if CONFIG_BT_MESH
+
+/**
+ * @brief Bluetooth Mesh Provisioning
+ * @defgroup bt_mesh_prov Bluetooth Mesh Provisioning
+ * @ingroup bt_mesh
+ * @{
+ */
+
+typedef enum {
+ BT_MESH_NO_OUTPUT = 0,
+ BT_MESH_BLINK = BIT(0),
+ BT_MESH_BEEP = BIT(1),
+ BT_MESH_VIBRATE = BIT(2),
+ BT_MESH_DISPLAY_NUMBER = BIT(3),
+ BT_MESH_DISPLAY_STRING = BIT(4),
+} bt_mesh_output_action_t;
+
+typedef enum {
+ BT_MESH_NO_INPUT = 0,
+ BT_MESH_PUSH = BIT(0),
+ BT_MESH_TWIST = BIT(1),
+ BT_MESH_ENTER_NUMBER = BIT(2),
+ BT_MESH_ENTER_STRING = BIT(3),
+} bt_mesh_input_action_t;
+
+typedef enum {
+ BT_MESH_PROV_ADV = BIT(0),
+ BT_MESH_PROV_GATT = BIT(1),
+} bt_mesh_prov_bearer_t;
+
+/** Provisioning properties & capabilities. */
+struct bt_mesh_prov {
+#if CONFIG_BT_MESH_NODE
+ /** The UUID that's used when advertising as unprovisioned */
+ const u8_t *uuid;
+
+ /** Flag indicates whether unprovisioned devices support OOB public key */
+ bool oob_pub_key;
+
+ /** @brief Set device OOB public key.
+ *
+ * This callback notifies the application to
+ * set OOB public key & private key pair.
+ */
+ void (*oob_pub_key_cb)(void);
+
+ /** Static OOB value */
+ const u8_t *static_val;
+ /** Static OOB value length */
+ u8_t static_val_len;
+
+ /** Maximum size of Output OOB supported */
+ u8_t output_size;
+ /** Supported Output OOB Actions */
+ u16_t output_actions;
+
+ /* Maximum size of Input OOB supported */
+ u8_t input_size;
+ /** Supported Input OOB Actions */
+ u16_t input_actions;
+
+ /** @brief Output of a number is requested.
+ *
+ * This callback notifies the application to
+ * output the given number using the given action.
+ *
+ * @param act Action for outputting the number.
+ * @param num Number to be out-put.
+ *
+ * @return Zero on success or negative error code otherwise
+ */
+ int (*output_number)(bt_mesh_output_action_t act, u32_t num);
+
+ /** @brief Output of a string is requested.
+ *
+ * This callback notifies the application to
+ * display the given string to the user.
+ *
+ * @param str String to be displayed.
+ *
+ * @return Zero on success or negative error code otherwise
+ */
+ int (*output_string)(const char *str);
+
+ /** @brief Input is requested.
+ *
+ * This callback notifies the application to request
+ * input from the user using the given action. The
+ * requested input will either be a string or a number, and
+ * the application needs to consequently call the
+ * bt_mesh_input_string() or bt_mesh_input_number() functions
+ * once the data has been acquired from the user.
+ *
+ * @param act Action for inputting data.
+ * @param num Maximum size of the in-put data.
+ *
+ * @return Zero on success or negative error code otherwise
+ */
+ int (*input)(bt_mesh_input_action_t act, u8_t size);
+
+ /** @brief Provisioning link has been opened.
+ *
+ * This callback notifies the application that a provisioning
+ * link has been opened on the given provisioning bearer.
+ *
+ * @param bearer Provisioning bearer.
+ */
+ void (*link_open)(bt_mesh_prov_bearer_t bearer);
+
+ /** @brief Provisioning link has been closed.
+ *
+ * This callback notifies the application that a provisioning
+ * link has been closed on the given provisioning bearer.
+ *
+ * @param bearer Provisioning bearer.
+ */
+ void (*link_close)(bt_mesh_prov_bearer_t bearer);
+
+ /** @brief Provisioning is complete.
+ *
+ * This callback notifies the application that provisioning has
+ * been successfully completed, and that the local node has been
+ * assigned the specified NetKeyIndex and primary element address.
+ *
+ * @param net_idx NetKeyIndex given during provisioning.
+ * @param addr Primary element address.
+ * @param flags Key Refresh & IV Update flags
+ * @param iv_index IV Index.
+ */
+ void (*complete)(u16_t net_idx, u16_t addr, u8_t flags, u32_t iv_index);
+
+ /** @brief Node has been reset.
+ *
+ * This callback notifies the application that the local node
+ * has been reset and needs to be reprovisioned. The node will
+ * not automatically advertise as unprovisioned, rather the
+ * bt_mesh_prov_enable() API needs to be called to enable
+ * unprovisioned advertising on one or more provisioning bearers.
+ */
+ void (*reset)(void);
+#endif /* CONFIG_BT_MESH_NODE */
+
+#if CONFIG_BT_MESH_PROVISIONER
+ /* Provisioner device uuid */
+ const u8_t *prov_uuid;
+
+ /*
+ * Primary element address of the provisioner.
+ * No need to initialize it for fast provisioning.
+ */
+ const u16_t prov_unicast_addr;
+
+ /*
+ * Starting unicast address going to assigned.
+ * No need to initialize it for fast provisioning.
+ */
+ u16_t prov_start_address;
+
+ /* Attention timer contained in Provisioning Invite */
+ u8_t prov_attention;
+
+ /* Provisioner provisioning Algorithm */
+ u8_t prov_algorithm;
+
+ /* Provisioner public key oob */
+ u8_t prov_pub_key_oob;
+
+ /** @brief Input is requested.
+ *
+ * This callback notifies the application that it should
+ * read device's public key with OOB
+ *
+ * @param link_idx: The provisioning link index
+ *
+ * @return Zero on success or negative error code otherwise
+ */
+ int (*prov_pub_key_oob_cb)(u8_t link_idx);
+
+ /* Provisioner static oob value */
+ u8_t *prov_static_oob_val;
+
+ /* Provisioner static oob value length */
+ u8_t prov_static_oob_len;
+
+ /** @brief Provisioner input a number read from device output
+ *
+ * This callback notifies the application that it should
+ * input the number given by the device.
+ *
+ * @param method: The OOB authentication method
+ * @param act: The output action of the device
+ * @param size: The output size of the device
+ * @param link_idx: The provisioning link index
+ *
+ * @return Zero on success or negative error code otherwise
+ */
+ int (*prov_input_num)(u8_t method, bt_mesh_output_action_t act, u8_t size, u8_t link_idx);
+
+ /** @brief Provisioner output a number to the device
+ *
+ * This callback notifies the application that it should
+ * output the number to the device.
+ *
+ * @param method: The OOB authentication method
+ * @param act: The input action of the device
+ * @param data: The input number/string of the device
+ * @param size: The input size of the device
+ * @param link_idx: The provisioning link index
+ *
+ * @return Zero on success or negative error code otherwise
+ */
+ int (*prov_output_num)(u8_t method, bt_mesh_input_action_t act, void *data, u8_t size, u8_t link_idx);
+
+ /*
+ * Key refresh and IV update flag.
+ * No need to initialize it for fast provisioning.
+ */
+ u8_t flags;
+
+ /*
+ * IV index. No need to initialize it for fast provisioning.
+ */
+ u32_t iv_index;
+
+ /** @brief Provisioner has opened a provisioning link.
+ *
+ * This callback notifies the application that a provisioning
+ * link has been opened on the given provisioning bearer.
+ *
+ * @param bearer Provisioning bearer.
+ */
+ void (*prov_link_open)(bt_mesh_prov_bearer_t bearer);
+
+ /** @brief Provisioner has closed a provisioning link.
+ *
+ * This callback notifies the application that a provisioning
+ * link has been closed on the given provisioning bearer.
+ *
+ * @param bearer Provisioning bearer.
+ * @param reason Provisioning link close reason(disconnect reason)
+ * 0xFF: disconnect due to provisioner_pb_gatt_disable()
+ */
+ void (*prov_link_close)(bt_mesh_prov_bearer_t bearer, u8_t reason);
+
+ /** @brief Provision one device is complete.
+ *
+ * This callback notifies the application that provisioner has
+ * successfully provisioned a device, and the node has been assigned
+ * the specified NetKeyIndex and primary element address.
+ *
+ * @param node_idx Node index within the node(provisioned device) queue.
+ * @param device_uuid Provisioned device uuid pointer.
+ * @param unicast_addr Provisioned device assigned unicast address.
+ * @param element_num Provisioned device element number.
+ * @param netkey_idx Provisioned device assigned netkey index.
+ */
+ void (*prov_complete)(int node_idx, const u8_t device_uuid[16],
+ u16_t unicast_addr, u8_t element_num,
+ u16_t netkey_idx);
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+};
+
+/** @brief Provide provisioning input OOB string.
+ *
+ * This is intended to be called after the bt_mesh_prov input callback
+ * has been called with BT_MESH_ENTER_STRING as the action.
+ *
+ * @param str String.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_input_string(const char *str);
+
+/** @brief Provide provisioning input OOB number.
+ *
+ * This is intended to be called after the bt_mesh_prov input callback
+ * has been called with BT_MESH_ENTER_NUMBER as the action.
+ *
+ * @param num Number.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_input_number(u32_t num);
+
+/** @brief Enable specific provisioning bearers
+ *
+ * Enable one or more provisioning bearers.
+ *
+ * @param Bit-wise OR of provisioning bearers.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_prov_enable(bt_mesh_prov_bearer_t bearers);
+
+/** @brief Disable specific provisioning bearers
+ *
+ * Disable one or more provisioning bearers.
+ *
+ * @param Bit-wise OR of provisioning bearers.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_prov_disable(bt_mesh_prov_bearer_t bearers);
+
+/** @brief Indicate whether provisioner is enabled
+ *
+ * @return true - enabled, false - disabled.
+ */
+bool bt_mesh_is_provisioner_en(void);
+
+/* The following API is for fast provisioning */
+
+#if CONFIG_BT_MESH_FAST_PROV
+
+/** @brief Change the device action
+ *
+ * @param[IN] action: role of device to be set
+ * 0x01 - enter, 0x02 - suspend, 0x03 - exit
+ *
+ * @return status
+ */
+u8_t bt_mesh_set_fast_prov_action(u8_t action);
+
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+#if CONFIG_BT_MESH_PROVISIONER
+
+/** @brief Provide provisioning input OOB string.
+ *
+ * This is intended to be called after the bt_mesh_prov input callback
+ * has been called with BT_MESH_ENTER_STRING as the action.
+ *
+ * @param str String.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_prov_input_string(const char *str);
+
+/** @brief Provide provisioning input OOB number.
+ *
+ * This is intended to be called after the bt_mesh_prov input callback
+ * has been called with BT_MESH_ENTER_NUMBER as the action.
+ *
+ * @param num Number.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_prov_input_number(u32_t num);
+
+/** @brief Enable specific provisioning bearers
+ *
+ * Enable one or more provisioning bearers.
+ *
+ * @param bearers Bit-wise OR of provisioning bearers.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_provisioner_enable(bt_mesh_prov_bearer_t bearers);
+
+/** @brief Disable specific provisioning bearers
+ *
+ * Disable one or more provisioning bearers.
+ *
+ * @param bearers Bit-wise OR of provisioning bearers.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_provisioner_disable(bt_mesh_prov_bearer_t bearers);
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+/**
+ * @}
+ */
+
+/**
+ * @brief Bluetooth Mesh
+ * @defgroup bt_mesh Bluetooth Mesh
+ * @ingroup bluetooth
+ * @{
+ */
+
+/* Primary Network Key index */
+#define BT_MESH_NET_PRIMARY 0x000
+
+#define BT_MESH_RELAY_DISABLED 0x00
+#define BT_MESH_RELAY_ENABLED 0x01
+#define BT_MESH_RELAY_NOT_SUPPORTED 0x02
+
+#define BT_MESH_BEACON_DISABLED 0x00
+#define BT_MESH_BEACON_ENABLED 0x01
+
+#define BT_MESH_GATT_PROXY_DISABLED 0x00
+#define BT_MESH_GATT_PROXY_ENABLED 0x01
+#define BT_MESH_GATT_PROXY_NOT_SUPPORTED 0x02
+
+#define BT_MESH_FRIEND_DISABLED 0x00
+#define BT_MESH_FRIEND_ENABLED 0x01
+#define BT_MESH_FRIEND_NOT_SUPPORTED 0x02
+
+#define BT_MESH_NODE_IDENTITY_STOPPED 0x00
+#define BT_MESH_NODE_IDENTITY_RUNNING 0x01
+#define BT_MESH_NODE_IDENTITY_NOT_SUPPORTED 0x02
+
+/* Features */
+#define BT_MESH_FEAT_RELAY BIT(0)
+#define BT_MESH_FEAT_PROXY BIT(1)
+#define BT_MESH_FEAT_FRIEND BIT(2)
+#define BT_MESH_FEAT_LOW_POWER BIT(3)
+#define BT_MESH_FEAT_SUPPORTED (BT_MESH_FEAT_RELAY | \
+ BT_MESH_FEAT_PROXY | \
+ BT_MESH_FEAT_FRIEND | \
+ BT_MESH_FEAT_LOW_POWER)
+
+/** @brief Initialize Mesh support
+ *
+ * After calling this API, the node will not automatically advertise as
+ * unprovisioned, rather the bt_mesh_prov_enable() API needs to be called
+ * to enable unprovisioned advertising on one or more provisioning bearers.
+ *
+ * @param prov Node provisioning information.
+ * @param comp Node Composition.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_init(const struct bt_mesh_prov *prov,
+ const struct bt_mesh_comp *comp);
+
+/** @brief Reset the state of the local Mesh node.
+ *
+ * Resets the state of the node, which means that it needs to be
+ * reprovisioned to become an active node in a Mesh network again.
+ *
+ * After calling this API, the node will not automatically advertise as
+ * unprovisioned, rather the bt_mesh_prov_enable() API needs to be called
+ * to enable unprovisioned advertising on one or more provisioning bearers.
+ *
+ */
+void bt_mesh_reset(void);
+
+/** @brief Provision the local Mesh Node.
+ *
+ * This API should normally not be used directly by the application. The
+ * only exception is for testing purposes where manual provisioning is
+ * desired without an actual external provisioner.
+ *
+ * @param net_key Network Key
+ * @param net_idx Network Key Index
+ * @param flags Provisioning Flags
+ * @param iv_index IV Index
+ * @param seq Sequence Number (0 if newly provisioned).
+ * @param addr Primary element address
+ * @param dev_key Device Key
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_provision(const u8_t net_key[16], u16_t net_idx,
+ u8_t flags, u32_t iv_index, u32_t seq,
+ u16_t addr, const u8_t dev_key[16]);
+
+/** @brief Toggle the IV Update test mode
+ *
+ * This API is only available if the IV Update test mode has been enabled
+ * in Kconfig. It is needed for passing most of the IV Update qualification
+ * test cases.
+ *
+ * @param enable true to enable IV Update test mode, false to disable it.
+ */
+void bt_mesh_iv_update_test(bool enable);
+
+/** @brief Toggle the IV Update state
+ *
+ * This API is only available if the IV Update test mode has been enabled
+ * in Kconfig. It is needed for passing most of the IV Update qualification
+ * test cases.
+ *
+ * @return true if IV Update In Progress state was entered, false otherwise.
+ */
+bool bt_mesh_iv_update(void);
+
+/** @brief Toggle the Low Power feature of the local device
+ *
+ * Enables or disables the Low Power feature of the local device. This is
+ * exposed as a run-time feature, since the device might want to change
+ * this e.g. based on being plugged into a stable power source or running
+ * from a battery power source.
+ *
+ * @param enable true to enable LPN functionality, false to disable it.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_lpn_set(bool enable);
+
+/** @brief Send out a Friend Poll message.
+ *
+ * Send a Friend Poll message to the Friend of this node. If there is no
+ * established Friendship the function will return an error.
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_lpn_poll(void);
+
+/** @brief Register a callback for Friendship changes.
+ *
+ * Registers a callback that will be called whenever Friendship gets
+ * established or is lost.
+ *
+ * @param cb Function to call when the Friendship status changes.
+ */
+void bt_mesh_lpn_set_cb(void (*cb)(u16_t friend_addr, bool established));
+
+/**
+ * @}
+ */
+
+#endif /* __BT_MESH_MAIN_H */
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_slist.h b/components/bt/ble_mesh/mesh_core/include/mesh_slist.h
new file mode 100644
index 0000000000..f1bab90cd6
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_slist.h
@@ -0,0 +1,470 @@
+/*
+ * Copyright (c) 2016 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+/**
+ * @file
+ *
+ * @brief Single-linked list implementation
+ *
+ * Single-linked list implementation using inline macros/functions.
+ * This API is not thread safe, and thus if a list is used across threads,
+ * calls to functions must be protected with synchronization primitives.
+ */
+
+#ifndef __SLIST_H__
+#define __SLIST_H__
+
+#include
+#include
+#include "mesh_util.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+
+struct _snode {
+ struct _snode *next;
+};
+
+typedef struct _snode sys_snode_t;
+
+struct _slist {
+ sys_snode_t *head;
+ sys_snode_t *tail;
+};
+
+typedef struct _slist sys_slist_t;
+
+/**
+ * @brief Provide the primitive to iterate on a list
+ * Note: the loop is unsafe and thus __sn should not be removed
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_SLIST_FOR_EACH_NODE(l, n) {
+ *
+ * }
+ *
+ * This and other SYS_SLIST_*() macros are not thread safe.
+ *
+ * @param __sl A pointer on a sys_slist_t to iterate on
+ * @param __sn A sys_snode_t pointer to peek each node of the list
+ */
+#define SYS_SLIST_FOR_EACH_NODE(__sl, __sn) \
+ for (__sn = sys_slist_peek_head(__sl); __sn; \
+ __sn = sys_slist_peek_next(__sn))
+
+/**
+ * @brief Provide the primitive to iterate on a list, from a node in the list
+ * Note: the loop is unsafe and thus __sn should not be removed
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_SLIST_ITERATE_FROM_NODE(l, n) {
+ *
+ * }
+ *
+ * Like SYS_SLIST_FOR_EACH_NODE(), but __dn already contains a node in the list
+ * where to start searching for the next entry from. If NULL, it starts from
+ * the head.
+ *
+ * This and other SYS_SLIST_*() macros are not thread safe.
+ *
+ * @param __sl A pointer on a sys_slist_t to iterate on
+ * @param __sn A sys_snode_t pointer to peek each node of the list
+ * it contains the starting node, or NULL to start from the head
+ */
+#define SYS_SLIST_ITERATE_FROM_NODE(__sl, __sn) \
+ for (__sn = __sn ? sys_slist_peek_next_no_check(__sn) \
+ : sys_slist_peek_head(__sl); \
+ __sn; \
+ __sn = sys_slist_peek_next(__sn))
+
+/**
+ * @brief Provide the primitive to safely iterate on a list
+ * Note: __sn can be removed, it will not break the loop.
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_SLIST_FOR_EACH_NODE_SAFE(l, n, s) {
+ *
+ * }
+ *
+ * This and other SYS_SLIST_*() macros are not thread safe.
+ *
+ * @param __sl A pointer on a sys_slist_t to iterate on
+ * @param __sn A sys_snode_t pointer to peek each node of the list
+ * @param __sns A sys_snode_t pointer for the loop to run safely
+ */
+#define SYS_SLIST_FOR_EACH_NODE_SAFE(__sl, __sn, __sns) \
+ for (__sn = sys_slist_peek_head(__sl), \
+ __sns = sys_slist_peek_next(__sn); \
+ __sn; __sn = __sns, \
+ __sns = sys_slist_peek_next(__sn))
+
+/*
+ * @brief Provide the primitive to resolve the container of a list node
+ * Note: it is safe to use with NULL pointer nodes
+ *
+ * @param __ln A pointer on a sys_node_t to get its container
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_node_t within the container struct
+ */
+#define SYS_SLIST_CONTAINER(__ln, __cn, __n) \
+ ((__ln) ? CONTAINER_OF((__ln), __typeof__(*(__cn)), __n) : NULL)
+/*
+ * @brief Provide the primitive to peek container of the list head
+ *
+ * @param __sl A pointer on a sys_slist_t to peek
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_node_t within the container struct
+ */
+#define SYS_SLIST_PEEK_HEAD_CONTAINER(__sl, __cn, __n) \
+ SYS_SLIST_CONTAINER(sys_slist_peek_head(__sl), __cn, __n)
+
+/*
+ * @brief Provide the primitive to peek container of the list tail
+ *
+ * @param __sl A pointer on a sys_slist_t to peek
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_node_t within the container struct
+ */
+#define SYS_SLIST_PEEK_TAIL_CONTAINER(__sl, __cn, __n) \
+ SYS_SLIST_CONTAINER(sys_slist_peek_tail(__sl), __cn, __n)
+
+/*
+ * @brief Provide the primitive to peek the next container
+ *
+ * @param __cn Container struct type pointer
+ * @param __n The field name of sys_node_t within the container struct
+ */
+
+#define SYS_SLIST_PEEK_NEXT_CONTAINER(__cn, __n) \
+ ((__cn) ? SYS_SLIST_CONTAINER(sys_slist_peek_next(&((__cn)->__n)), \
+ __cn, __n) : NULL)
+
+/**
+ * @brief Provide the primitive to iterate on a list under a container
+ * Note: the loop is unsafe and thus __cn should not be detached
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_SLIST_FOR_EACH_CONTAINER(l, c, n) {
+ *
+ * }
+ *
+ * @param __sl A pointer on a sys_slist_t to iterate on
+ * @param __cn A pointer to peek each entry of the list
+ * @param __n The field name of sys_node_t within the container struct
+ */
+#define SYS_SLIST_FOR_EACH_CONTAINER(__sl, __cn, __n) \
+ for (__cn = SYS_SLIST_PEEK_HEAD_CONTAINER(__sl, __cn, __n); __cn; \
+ __cn = SYS_SLIST_PEEK_NEXT_CONTAINER(__cn, __n))
+
+/**
+ * @brief Provide the primitive to safely iterate on a list under a container
+ * Note: __cn can be detached, it will not break the loop.
+ *
+ * User _MUST_ add the loop statement curly braces enclosing its own code:
+ *
+ * SYS_SLIST_FOR_EACH_NODE_SAFE(l, c, cn, n) {
+ *
+ * }
+ *
+ * @param __sl A pointer on a sys_slist_t to iterate on
+ * @param __cn A pointer to peek each entry of the list
+ * @param __cns A pointer for the loop to run safely
+ * @param __n The field name of sys_node_t within the container struct
+ */
+#define SYS_SLIST_FOR_EACH_CONTAINER_SAFE(__sl, __cn, __cns, __n) \
+ for (__cn = SYS_SLIST_PEEK_HEAD_CONTAINER(__sl, __cn, __n), \
+ __cns = SYS_SLIST_PEEK_NEXT_CONTAINER(__cn, __n); __cn; \
+ __cn = __cns, __cns = SYS_SLIST_PEEK_NEXT_CONTAINER(__cn, __n))
+
+/**
+ * @brief Initialize a list
+ *
+ * @param list A pointer on the list to initialize
+ */
+static inline void sys_slist_init(sys_slist_t *list)
+{
+ list->head = NULL;
+ list->tail = NULL;
+}
+
+#define SYS_SLIST_STATIC_INIT(ptr_to_list) {NULL, NULL}
+
+/**
+ * @brief Test if the given list is empty
+ *
+ * @param list A pointer on the list to test
+ *
+ * @return a boolean, true if it's empty, false otherwise
+ */
+static inline bool sys_slist_is_empty(sys_slist_t *list)
+{
+ return (!list->head);
+}
+
+/**
+ * @brief Peek the first node from the list
+ *
+ * @param list A point on the list to peek the first node from
+ *
+ * @return A pointer on the first node of the list (or NULL if none)
+ */
+static inline sys_snode_t *sys_slist_peek_head(sys_slist_t *list)
+{
+ return list->head;
+}
+
+/**
+ * @brief Peek the last node from the list
+ *
+ * @param list A point on the list to peek the last node from
+ *
+ * @return A pointer on the last node of the list (or NULL if none)
+ */
+static inline sys_snode_t *sys_slist_peek_tail(sys_slist_t *list)
+{
+ return list->tail;
+}
+
+/**
+ * @brief Peek the next node from current node, node is not NULL
+ *
+ * Faster then sys_slist_peek_next() if node is known not to be NULL.
+ *
+ * @param node A pointer on the node where to peek the next node
+ *
+ * @return a pointer on the next node (or NULL if none)
+ */
+static inline sys_snode_t *sys_slist_peek_next_no_check(sys_snode_t *node)
+{
+ return node->next;
+}
+
+/**
+ * @brief Peek the next node from current node
+ *
+ * @param node A pointer on the node where to peek the next node
+ *
+ * @return a pointer on the next node (or NULL if none)
+ */
+static inline sys_snode_t *sys_slist_peek_next(sys_snode_t *node)
+{
+ return node ? sys_slist_peek_next_no_check(node) : NULL;
+}
+
+/**
+ * @brief Prepend a node to the given list
+ *
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ * @param node A pointer on the node to prepend
+ */
+static inline void sys_slist_prepend(sys_slist_t *list,
+ sys_snode_t *node)
+{
+ node->next = list->head;
+ list->head = node;
+
+ if (!list->tail) {
+ list->tail = list->head;
+ }
+}
+
+/**
+ * @brief Append a node to the given list
+ *
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ * @param node A pointer on the node to append
+ */
+static inline void sys_slist_append(sys_slist_t *list,
+ sys_snode_t *node)
+{
+ node->next = NULL;
+
+ if (!list->tail) {
+ list->tail = node;
+ list->head = node;
+ } else {
+ list->tail->next = node;
+ list->tail = node;
+ }
+}
+
+/**
+ * @brief Append a list to the given list
+ *
+ * Append a singly-linked, NULL-terminated list consisting of nodes containing
+ * the pointer to the next node as the first element of a node, to @a list.
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ * @param head A pointer to the first element of the list to append
+ * @param tail A pointer to the last element of the list to append
+ */
+static inline void sys_slist_append_list(sys_slist_t *list,
+ void *head, void *tail)
+{
+ if (!list->tail) {
+ list->head = (sys_snode_t *)head;
+ list->tail = (sys_snode_t *)tail;
+ } else {
+ list->tail->next = (sys_snode_t *)head;
+ list->tail = (sys_snode_t *)tail;
+ }
+}
+
+/**
+ * @brief merge two slists, appending the second one to the first
+ *
+ * When the operation is completed, the appending list is empty.
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ * @param list_to_append A pointer to the list to append.
+ */
+static inline void sys_slist_merge_slist(sys_slist_t *list,
+ sys_slist_t *list_to_append)
+{
+ sys_slist_append_list(list, list_to_append->head,
+ list_to_append->tail);
+ sys_slist_init(list_to_append);
+}
+
+/**
+ * @brief Insert a node to the given list
+ *
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ * @param prev A pointer on the previous node
+ * @param node A pointer on the node to insert
+ */
+static inline void sys_slist_insert(sys_slist_t *list,
+ sys_snode_t *prev,
+ sys_snode_t *node)
+{
+ if (!prev) {
+ sys_slist_prepend(list, node);
+ } else if (!prev->next) {
+ sys_slist_append(list, node);
+ } else {
+ node->next = prev->next;
+ prev->next = node;
+ }
+}
+
+/**
+ * @brief Fetch and remove the first node of the given list
+ *
+ * List must be known to be non-empty.
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ *
+ * @return A pointer to the first node of the list
+ */
+static inline sys_snode_t *sys_slist_get_not_empty(sys_slist_t *list)
+{
+ sys_snode_t *node = list->head;
+
+ list->head = node->next;
+ if (list->tail == node) {
+ list->tail = list->head;
+ }
+
+ return node;
+}
+
+/**
+ * @brief Fetch and remove the first node of the given list
+ *
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ *
+ * @return A pointer to the first node of the list (or NULL if empty)
+ */
+static inline sys_snode_t *sys_slist_get(sys_slist_t *list)
+{
+ return sys_slist_is_empty(list) ? NULL : sys_slist_get_not_empty(list);
+}
+
+/**
+ * @brief Remove a node
+ *
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ * @param prev_node A pointer on the previous node
+ * (can be NULL, which means the node is the list's head)
+ * @param node A pointer on the node to remove
+ */
+static inline void sys_slist_remove(sys_slist_t *list,
+ sys_snode_t *prev_node,
+ sys_snode_t *node)
+{
+ if (!prev_node) {
+ list->head = node->next;
+
+ /* Was node also the tail? */
+ if (list->tail == node) {
+ list->tail = list->head;
+ }
+ } else {
+ prev_node->next = node->next;
+
+ /* Was node the tail? */
+ if (list->tail == node) {
+ list->tail = prev_node;
+ }
+ }
+
+ node->next = NULL;
+}
+
+/**
+ * @brief Find and remove a node from a list
+ *
+ * This and other sys_slist_*() functions are not thread safe.
+ *
+ * @param list A pointer on the list to affect
+ * @param node A pointer on the node to remove from the list
+ *
+ * @return true if node was removed
+ */
+static inline bool sys_slist_find_and_remove(sys_slist_t *list,
+ sys_snode_t *node)
+{
+ sys_snode_t *prev = NULL;
+ sys_snode_t *test;
+
+ SYS_SLIST_FOR_EACH_NODE(list, test) {
+ if (test == node) {
+ sys_slist_remove(list, prev, node);
+ return true;
+ }
+
+ prev = test;
+ }
+
+ return false;
+}
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __SLIST_H__ */
+
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_trace.h b/components/bt/ble_mesh/mesh_core/include/mesh_trace.h
new file mode 100644
index 0000000000..a498863323
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_trace.h
@@ -0,0 +1,114 @@
+/*
+ * Copyright (c) 2017 Nordic Semiconductor ASA
+ * Copyright (c) 2015-2016 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "esp_log.h"
+#include "sdkconfig.h"
+
+/* Define common tracing for all */
+#ifndef LOG_LEVEL_ERROR
+#define LOG_LEVEL_ERROR 1
+#endif /* LOG_LEVEL_ERROR */
+
+#ifndef LOG_LEVEL_WARN
+#define LOG_LEVEL_WARN 2
+#endif /* LOG_LEVEL_WARN */
+
+#ifndef LOG_LEVEL_INFO
+#define LOG_LEVEL_INFO 3
+#endif /* LOG_LEVEL_INFO */
+
+#ifndef LOG_LEVEL_DEBUG
+#define LOG_LEVEL_DEBUG 4
+#endif /* LOG_LEVEL_DEBUG */
+
+#ifndef LOG_LEVEL_VERBOSE
+#define LOG_LEVEL_VERBOSE 5
+#endif /*LOG_LEVEL_VERBOSE */
+
+#ifdef CONFIG_BT_MESH_STACK_INITIAL_TRACE_LEVEL
+#define MESH_LOG_LEVEL CONFIG_BT_MESH_STACK_INITIAL_TRACE_LEVEL
+#else
+#define MESH_LOG_LEVEL LOG_LEVEL_WARN
+#endif
+
+#ifdef CONFIG_BT_MESH_NET_BUF_TRACE_LEVEL
+#define NET_BUF_LOG_LEVEL CONFIG_BT_MESH_NET_BUF_TRACE_LEVEL
+#else
+#define NET_BUF_LOG_LEVEL LOG_LEVEL_WARN
+#endif
+
+
+#define MESH_TRACE_TAG "BLE_MESH"
+
+#if (LOG_LOCAL_LEVEL >= 4)
+#define BT_MESH_LOG_LOCAL_LEVEL_MAPPING (LOG_LOCAL_LEVEL + 1)
+#else
+#define BT_MESH_LOG_LOCAL_LEVEL_MAPPING LOG_LOCAL_LEVEL
+#endif
+
+#ifndef MAX
+#define MAX(a, b) ((a) > (b) ? (a) : (b))
+#endif /* MAX(a, b) */
+#define BT_MESH_LOG_LEVEL_CHECK(LAYER, LEVEL) (MAX(LAYER##_LOG_LEVEL, BT_MESH_LOG_LOCAL_LEVEL_MAPPING) >= LOG_LEVEL_##LEVEL)
+
+#define BT_MESH_PRINT_E(tag, format, ...) {esp_log_write(ESP_LOG_ERROR, tag, LOG_FORMAT(E, format), esp_log_timestamp(), tag, ##__VA_ARGS__); }
+#define BT_MESH_PRINT_W(tag, format, ...) {esp_log_write(ESP_LOG_WARN, tag, LOG_FORMAT(W, format), esp_log_timestamp(), tag, ##__VA_ARGS__); }
+#define BT_MESH_PRINT_I(tag, format, ...) {esp_log_write(ESP_LOG_INFO, tag, LOG_FORMAT(I, format), esp_log_timestamp(), tag, ##__VA_ARGS__); }
+#define BT_MESH_PRINT_D(tag, format, ...) {esp_log_write(ESP_LOG_DEBUG, tag, LOG_FORMAT(D, format), esp_log_timestamp(), tag, ##__VA_ARGS__); }
+#define BT_MESH_PRINT_V(tag, format, ...) {esp_log_write(ESP_LOG_VERBOSE, tag, LOG_FORMAT(V, format), esp_log_timestamp(), tag, ##__VA_ARGS__); }
+
+#ifndef __ASSERT_NO_MSG
+#define __ASSERT_NO_MSG(x) do { if (!(x)) BT_MESH_PRINT_E(MESH_TRACE_TAG, "bt mesh error %s %u\n", __FILE__, __LINE__); } while (0)
+#endif
+
+#if !CONFIG_BT_MESH_NO_LOG
+#define BT_ERR(fmt, args...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_ERROR) && BT_MESH_LOG_LEVEL_CHECK(MESH, ERROR)) BT_MESH_PRINT_E(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#define BT_WARN(fmt, args...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_WARN) && BT_MESH_LOG_LEVEL_CHECK(MESH, WARN)) BT_MESH_PRINT_W(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#define BT_INFO(fmt, args...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_INFO) && BT_MESH_LOG_LEVEL_CHECK(MESH, INFO)) BT_MESH_PRINT_I(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#define BT_DBG(fmt, args...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_DEBUG) && BT_MESH_LOG_LEVEL_CHECK(MESH, DEBUG)) BT_MESH_PRINT_D(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#else /* #if !CONFIG_BT_STACK_NO_LOG */
+#define BT_ERR(fmt, args...)
+#define BT_WARN(fmt, args...)
+#define BT_INFO(fmt, args...)
+#define BT_DBG(fmt, args...)
+#endif /* #if !CONFIG_BT_STACK_NO_LOG */
+
+#if defined(CONFIG_NET_BUF_LOG) && (!CONFIG_BT_MESH_NO_LOG)
+#define NET_BUF_DBG(fmt, ...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_ERROR) && BT_MESH_LOG_LEVEL_CHECK(NET_BUF, DEBUG)) BT_MESH_PRINT_D(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#define NET_BUF_ERR(fmt, ...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_ERROR) && BT_MESH_LOG_LEVEL_CHECK(NET_BUF, ERROR)) BT_MESH_PRINT_E(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#define NET_BUF_WARN(fmt, ...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_ERROR) && BT_MESH_LOG_LEVEL_CHECK(NET_BUF, WARN)) BT_MESH_PRINT_W(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#define NET_BUF_INFO(fmt, ...) do {if ((MESH_LOG_LEVEL >= LOG_LEVEL_ERROR) && BT_MESH_LOG_LEVEL_CHECK(NET_BUF, INFO)) BT_MESH_PRINT_I(MESH_TRACE_TAG, fmt, ## args);} while(0)
+#define NET_BUF_ASSERT(cond) do { if (!(cond)) { \
+ NET_BUF_ERR("assert: '" #cond "' failed"); \
+ } } while (0)
+#else
+#define NET_BUF_DBG(fmt, ...)
+#define NET_BUF_ERR(fmt, ...)
+#define NET_BUF_WARN(fmt, ...)
+#define NET_BUF_INFO(fmt, ...)
+#define NET_BUF_ASSERT(cond)
+#endif /* CONFIG_NET_BUF_LOG */
+
+#if defined(CONFIG_NET_BUF_SIMPLE_LOG) && (!CONFIG_BT_MESH_NO_LOG)
+#define NET_BUF_SIMPLE_DBG(fmt, ...) do {if (MESH_LOG_LEVEL >= LOG_LEVEL_DEBUG) esp_log_write(ESP_LOG_ERROR, MESH_TRACE_TAG, LOG_FORMAT(D, format), esp_log_timestamp(), "BT_LOG", ##__VA_ARGS__);} while(0)
+#define NET_BUF_SIMPLE_ERR(fmt, ...) do {if (MESH_LOG_LEVEL >= LOG_LEVEL_ERROR) esp_log_write(ESP_LOG_ERROR, MESH_TRACE_TAG, LOG_FORMAT(E, format), esp_log_timestamp(), "BT_LOG", ##__VA_ARGS__);} while(0)
+#define NET_BUF_SIMPLE_WARN(fmt, ...) do {if (MESH_LOG_LEVEL >= LOG_LEVEL_WARN) esp_log_write(ESP_LOG_ERROR, MESH_TRACE_TAG, LOG_FORMAT(W, format), esp_log_timestamp(), "BT_LOG", ##__VA_ARGS__);} while(0)
+#define NET_BUF_SIMPLE_INFO(fmt, ...) do {if (MESH_LOG_LEVEL >= LOG_LEVEL_INFO) esp_log_write(ESP_LOG_ERROR, MESH_TRACE_TAG, LOG_FORMAT(I, format), esp_log_timestamp(), "BT_LOG", ##__VA_ARGS__);} while(0)
+#define NET_BUF_SIMPLE_ASSERT(cond) assert(cond != NULL)
+#else /* CONFIG_NET_BUF_SIMPLE_LOG */
+#define NET_BUF_SIMPLE_DBG(fmt, ...)
+#define NET_BUF_SIMPLE_ERR(fmt, ...)
+#define NET_BUF_SIMPLE_WARN(fmt, ...)
+#define NET_BUF_SIMPLE_INFO(fmt, ...)
+#define NET_BUF_SIMPLE_ASSERT(cond)
+
+#endif /* CONFIG_NET_BUF_SIMPLE_LOG */
+
+
+#define printk ets_printf
+
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_types.h b/components/bt/ble_mesh/mesh_core/include/mesh_types.h
new file mode 100644
index 0000000000..a3ddf774d2
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_types.h
@@ -0,0 +1,40 @@
+/*
+ * Copyright (c) 2017 Linaro Limited
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __Z_TYPES_H__
+#define __Z_TYPES_H__
+
+#include
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef signed char s8_t;
+typedef signed short s16_t;
+typedef signed int s32_t;
+typedef signed long long s64_t;
+
+typedef unsigned char u8_t;
+typedef unsigned short u16_t;
+typedef unsigned int u32_t;
+typedef unsigned long long u64_t;
+
+typedef int atomic_t;
+
+#ifndef bool
+#define bool int8_t
+#define false 0 ///< XOS definition of 'false'
+#define true 1 ///< XOS definition of 'true'
+#endif
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __Z_TYPES_H__ */
+
diff --git a/components/bt/ble_mesh/mesh_core/include/mesh_util.h b/components/bt/ble_mesh/mesh_core/include/mesh_util.h
new file mode 100644
index 0000000000..a15f23d2b0
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/mesh_util.h
@@ -0,0 +1,629 @@
+/*
+ * Copyright (c) 2011-2014, Wind River Systems, Inc.
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+/**
+ * @file
+ * @brief Misc utilities
+ *
+ * Misc utilities usable by the kernel and application code.
+ */
+
+#ifndef _UTIL__H_
+#define _UTIL__H_
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#ifndef _ASMLANGUAGE
+
+#include
+#include "mesh_types.h"
+#include "mesh_trace.h"
+#include "soc/soc.h"
+
+/* Helper to pass a int as a pointer or vice-versa.
+ * Those are available for 32 bits architectures:
+ */
+#define POINTER_TO_UINT(x) ((u32_t) (x))
+#define UINT_TO_POINTER(x) ((void *) (x))
+#define POINTER_TO_INT(x) ((s32_t) (x))
+#define INT_TO_POINTER(x) ((void *) (x))
+
+/* Evaluates to 0 if cond is true-ish; compile error otherwise */
+#define ZERO_OR_COMPILE_ERROR(cond) ((int) sizeof(char[1 - 2 * !(cond)]) - 1)
+
+/* Evaluates to 0 if array is an array; compile error if not array (e.g.
+ * pointer)
+ */
+#define IS_ARRAY(array) \
+ ZERO_OR_COMPILE_ERROR( \
+ !__builtin_types_compatible_p(__typeof__(array), \
+ __typeof__(&(array)[0])))
+
+/* Evaluates to number of elements in an array; compile error if not
+ * an array (e.g. pointer)
+ */
+#define ARRAY_SIZE(array) \
+ ((unsigned long) (IS_ARRAY(array) + \
+ (sizeof(array) / sizeof((array)[0]))))
+
+/* Evaluates to 1 if ptr is part of array, 0 otherwise; compile error if
+ * "array" argument is not an array (e.g. "ptr" and "array" mixed up)
+ */
+#define PART_OF_ARRAY(array, ptr) \
+ ((ptr) && ((ptr) >= &array[0] && (ptr) < &array[ARRAY_SIZE(array)]))
+
+#define CONTAINER_OF(ptr, type, field) \
+ ((type *)(((char *)(ptr)) - offsetof(type, field)))
+
+/* round "x" up/down to next multiple of "align" (which must be a power of 2) */
+#define ROUND_UP(x, align) \
+ (((unsigned long)(x) + ((unsigned long)align - 1)) & \
+ ~((unsigned long)align - 1))
+#define ROUND_DOWN(x, align) ((unsigned long)(x) & ~((unsigned long)align - 1))
+
+#define ceiling_fraction(numerator, divider) \
+ (((numerator) + ((divider) - 1)) / (divider))
+
+/* Internal helpers only used by the sys_* APIs further below */
+#define __bswap_16(x) ((u16_t) ((((x) >> 8) & 0xff) | (((x) & 0xff) << 8)))
+#define __bswap_32(x) ((u32_t) ((((x) >> 24) & 0xff) | \
+ (((x) >> 8) & 0xff00) | \
+ (((x) & 0xff00) << 8) | \
+ (((x) & 0xff) << 24)))
+#define __bswap_64(x) ((u64_t) ((((x) >> 56) & 0xff) | \
+ (((x) >> 40) & 0xff00) | \
+ (((x) >> 24) & 0xff0000) | \
+ (((x) >> 8) & 0xff000000) | \
+ (((x) & 0xff000000) << 8) | \
+ (((x) & 0xff0000) << 24) | \
+ (((x) & 0xff00) << 40) | \
+ (((x) & 0xff) << 56)))
+
+#define sys_le16_to_cpu(val) (val)
+#define sys_cpu_to_le16(val) (val)
+#define sys_be16_to_cpu(val) __bswap_16(val)
+#define sys_cpu_to_be16(val) __bswap_16(val)
+#define sys_le32_to_cpu(val) (val)
+#define sys_cpu_to_le32(val) (val)
+#define sys_le64_to_cpu(val) (val)
+#define sys_cpu_to_le64(val) (val)
+#define sys_be32_to_cpu(val) __bswap_32(val)
+#define sys_cpu_to_be32(val) __bswap_32(val)
+#define sys_be64_to_cpu(val) __bswap_64(val)
+#define sys_cpu_to_be64(val) __bswap_64(val)
+
+
+
+#ifdef INLINED
+#define INLINE inline
+#else
+#define INLINE
+#endif
+
+#define NOINIT
+
+#ifndef max
+#define max(a, b) (((a) > (b)) ? (a) : (b))
+#endif
+
+#ifndef min
+#define min(a, b) (((a) < (b)) ? (a) : (b))
+#endif
+
+static inline int is_power_of_two(unsigned int x)
+{
+ return (x != 0) && !(x & (x - 1));
+}
+
+static inline s64_t arithmetic_shift_right(s64_t value, u8_t shift)
+{
+ s64_t sign_ext;
+
+ if (shift == 0) {
+ return value;
+ }
+
+ /* extract sign bit */
+ sign_ext = (value >> 63) & 1;
+
+ /* make all bits of sign_ext be the same as the value's sign bit */
+ sign_ext = -sign_ext;
+
+ /* shift value and fill opened bit positions with sign bit */
+ return (value >> shift) | (sign_ext << (64 - shift));
+}
+
+#endif /* !_ASMLANGUAGE */
+
+/* KB, MB, GB */
+#define KB(x) ((x) << 10)
+#define MB(x) (KB(x) << 10)
+#define GB(x) (MB(x) << 10)
+
+/* KHZ, MHZ */
+#define KHZ(x) ((x) * 1000)
+#define MHZ(x) (KHZ(x) * 1000)
+
+#ifndef BIT
+#define BIT(n) (1UL << (n))
+#endif
+
+#define BIT_MASK(n) (BIT(n) - 1)
+
+/**
+ * @brief Check for macro definition in compiler-visible expressions
+ *
+ * This trick was pioneered in Linux as the config_enabled() macro.
+ * The madness has the effect of taking a macro value that may be
+ * defined to "1" (e.g. CONFIG_MYFEATURE), or may not be defined at
+ * all and turning it into a literal expression that can be used at
+ * "runtime". That is, it works similarly to
+ * "defined(CONFIG_MYFEATURE)" does except that it is an expansion
+ * that can exist in a standard expression and be seen by the compiler
+ * and optimizer. Thus much ifdef usage can be replaced with cleaner
+ * expressions like:
+ *
+ * if (IS_ENABLED(CONFIG_MYFEATURE))
+ * myfeature_enable();
+ *
+ * INTERNAL
+ * First pass just to expand any existing macros, we need the macro
+ * value to be e.g. a literal "1" at expansion time in the next macro,
+ * not "(1)", etc... Standard recursive expansion does not work.
+ */
+#define IS_ENABLED(config_macro) _IS_ENABLED1(config_macro)
+
+/* Now stick on a "_XXXX" prefix, it will now be "_XXXX1" if config_macro
+ * is "1", or just "_XXXX" if it's undefined.
+ * ENABLED: _IS_ENABLED2(_XXXX1)
+ * DISABLED _IS_ENABLED2(_XXXX)
+ */
+#define _IS_ENABLED1(config_macro) _IS_ENABLED2(_XXXX##config_macro)
+
+/* Here's the core trick, we map "_XXXX1" to "_YYYY," (i.e. a string
+ * with a trailing comma), so it has the effect of making this a
+ * two-argument tuple to the preprocessor only in the case where the
+ * value is defined to "1"
+ * ENABLED: _YYYY, <--- note comma!
+ * DISABLED: _XXXX
+ */
+#define _XXXX1 _YYYY,
+
+/* Then we append an extra argument to fool the gcc preprocessor into
+ * accepting it as a varargs macro.
+ * arg1 arg2 arg3
+ * ENABLED: _IS_ENABLED3(_YYYY, 1, 0)
+ * DISABLED _IS_ENABLED3(_XXXX 1, 0)
+ */
+#define _IS_ENABLED2(one_or_two_args) _IS_ENABLED3(one_or_two_args 1, 0)
+
+/* And our second argument is thus now cooked to be 1 in the case
+ * where the value is defined to 1, and 0 if not:
+ */
+#define _IS_ENABLED3(ignore_this, val, ...) val
+
+/**
+ * Macros for doing code-generation with the preprocessor.
+ *
+ * Generally it is better to generate code with the preprocessor than
+ * to copy-paste code or to generate code with the build system /
+ * python script's etc.
+ *
+ * http://stackoverflow.com/a/12540675
+ */
+#define UTIL_EMPTY(...)
+#define UTIL_DEFER(...) __VA_ARGS__ UTIL_EMPTY()
+#define UTIL_OBSTRUCT(...) __VA_ARGS__ UTIL_DEFER(UTIL_EMPTY)()
+#define UTIL_EXPAND(...) __VA_ARGS__
+
+#define UTIL_EVAL(...) UTIL_EVAL1(UTIL_EVAL1(UTIL_EVAL1(__VA_ARGS__)))
+#define UTIL_EVAL1(...) UTIL_EVAL2(UTIL_EVAL2(UTIL_EVAL2(__VA_ARGS__)))
+#define UTIL_EVAL2(...) UTIL_EVAL3(UTIL_EVAL3(UTIL_EVAL3(__VA_ARGS__)))
+#define UTIL_EVAL3(...) UTIL_EVAL4(UTIL_EVAL4(UTIL_EVAL4(__VA_ARGS__)))
+#define UTIL_EVAL4(...) UTIL_EVAL5(UTIL_EVAL5(UTIL_EVAL5(__VA_ARGS__)))
+#define UTIL_EVAL5(...) __VA_ARGS__
+
+#define UTIL_CAT(a, ...) UTIL_PRIMITIVE_CAT(a, __VA_ARGS__)
+#define UTIL_PRIMITIVE_CAT(a, ...) a##__VA_ARGS__
+
+#define UTIL_INC(x) UTIL_PRIMITIVE_CAT(UTIL_INC_, x)
+#define UTIL_INC_0 1
+#define UTIL_INC_1 2
+#define UTIL_INC_2 3
+#define UTIL_INC_3 4
+#define UTIL_INC_4 5
+#define UTIL_INC_5 6
+#define UTIL_INC_6 7
+#define UTIL_INC_7 8
+#define UTIL_INC_8 9
+#define UTIL_INC_9 10
+#define UTIL_INC_10 11
+#define UTIL_INC_11 12
+#define UTIL_INC_12 13
+#define UTIL_INC_13 14
+#define UTIL_INC_14 15
+#define UTIL_INC_15 16
+#define UTIL_INC_16 17
+#define UTIL_INC_17 18
+#define UTIL_INC_18 19
+#define UTIL_INC_19 19
+
+#define UTIL_DEC(x) UTIL_PRIMITIVE_CAT(UTIL_DEC_, x)
+#define UTIL_DEC_0 0
+#define UTIL_DEC_1 0
+#define UTIL_DEC_2 1
+#define UTIL_DEC_3 2
+#define UTIL_DEC_4 3
+#define UTIL_DEC_5 4
+#define UTIL_DEC_6 5
+#define UTIL_DEC_7 6
+#define UTIL_DEC_8 7
+#define UTIL_DEC_9 8
+#define UTIL_DEC_10 9
+#define UTIL_DEC_11 10
+#define UTIL_DEC_12 11
+#define UTIL_DEC_13 12
+#define UTIL_DEC_14 13
+#define UTIL_DEC_15 14
+#define UTIL_DEC_16 15
+#define UTIL_DEC_17 16
+#define UTIL_DEC_18 17
+#define UTIL_DEC_19 18
+
+#define UTIL_CHECK_N(x, n, ...) n
+#define UTIL_CHECK(...) UTIL_CHECK_N(__VA_ARGS__, 0,)
+
+#define UTIL_NOT(x) UTIL_CHECK(UTIL_PRIMITIVE_CAT(UTIL_NOT_, x))
+#define UTIL_NOT_0 ~, 1,
+
+#define UTIL_COMPL(b) UTIL_PRIMITIVE_CAT(UTIL_COMPL_, b)
+#define UTIL_COMPL_0 1
+#define UTIL_COMPL_1 0
+
+#define UTIL_BOOL(x) UTIL_COMPL(UTIL_NOT(x))
+
+#define UTIL_IIF(c) UTIL_PRIMITIVE_CAT(UTIL_IIF_, c)
+#define UTIL_IIF_0(t, ...) __VA_ARGS__
+#define UTIL_IIF_1(t, ...) t
+
+#define UTIL_IF(c) UTIL_IIF(UTIL_BOOL(c))
+
+#define UTIL_EAT(...)
+#define UTIL_EXPAND(...) __VA_ARGS__
+#define UTIL_WHEN(c) UTIL_IF(c)(UTIL_EXPAND, UTIL_EAT)
+
+#define UTIL_REPEAT(count, macro, ...) \
+ UTIL_WHEN(count) \
+ ( \
+ UTIL_OBSTRUCT(UTIL_REPEAT_INDIRECT) () \
+ ( \
+ UTIL_DEC(count), macro, __VA_ARGS__ \
+ ) \
+ UTIL_OBSTRUCT(macro) \
+ ( \
+ UTIL_DEC(count), __VA_ARGS__ \
+ ) \
+ )
+#define UTIL_REPEAT_INDIRECT() UTIL_REPEAT
+
+#define _STRINGIFY(x) #x
+#define STRINGIFY(s) _STRINGIFY(s)
+
+/* ESP Toolchain doesn't support section */
+#define ___in_section(a, b, c)
+#define __in_section(a, b, c) ___in_section(a, b, c)
+
+#define __in_section_unique(seg) ___in_section(seg, __FILE__, __COUNTER__)
+
+
+/* ESP Toolchain doesn't support section */
+#define __noinit
+
+#define popcount(x) __builtin_popcount(x)
+
+
+/**
+ *
+ * @brief find most significant bit set in a 32-bit word
+ *
+ * This routine finds the first bit set starting from the most significant bit
+ * in the argument passed in and returns the index of that bit. Bits are
+ * numbered starting at 1 from the least significant bit. A return value of
+ * zero indicates that the value passed is zero.
+ *
+ * @return most significant bit set, 0 if @a op is 0
+ */
+
+#if defined(__GNUC__)
+static inline unsigned int find_msb_set(u32_t op)
+{
+ if (!op) {
+ return 0;
+ }
+ return 32 - __builtin_clz(op);
+}
+#endif
+
+/**
+ *
+ * @brief find least significant bit set in a 32-bit word
+ *
+ * This routine finds the first bit set starting from the least significant bit
+ * in the argument passed in and returns the index of that bit. Bits are
+ * numbered starting at 1 from the least significant bit. A return value of
+ * zero indicates that the value passed is zero.
+ *
+ * @return least significant bit set, 0 if @a op is 0
+ */
+
+#if defined(__GNUC__)
+static inline unsigned int find_lsb_set(u32_t op)
+{
+ return __builtin_ffs(op);
+}
+#endif
+
+#define __ASSERT(test, fmt, ...) \
+ do { \
+ if (!(test)) { \
+ printk("ASSERTION FAIL [%s] @ %s:%d:\n\t", \
+ _STRINGIFY(test), \
+ __FILE__, \
+ __LINE__); \
+ printk(fmt, ##__VA_ARGS__); \
+ for (;;) \
+ ; /* spin thread */ \
+ } \
+ } while ((0))
+
+/**
+ * Generates a sequence of code.
+ * Useful for generating code like;
+ *
+ * NRF_PWM0, NRF_PWM1, NRF_PWM2,
+ *
+ * @arg LEN: The length of the sequence. Must be defined and less than
+ * 20.
+ *
+ * @arg F(i, F_ARG): A macro function that accepts two arguments.
+ * F is called repeatedly, the first argument
+ * is the index in the sequence, and the second argument is the third
+ * argument given to UTIL_LISTIFY.
+ *
+ * Example:
+ *
+ * \#define FOO(i, _) NRF_PWM ## i ,
+ * { UTIL_LISTIFY(PWM_COUNT, FOO) }
+ * // The above two lines will generate the below:
+ * { NRF_PWM0 , NRF_PWM1 , }
+ *
+ * @note Calling UTIL_LISTIFY with undefined arguments has undefined
+ * behaviour.
+ */
+#define UTIL_LISTIFY(LEN, F, F_ARG) UTIL_EVAL(UTIL_REPEAT(LEN, F, F_ARG))
+
+/**
+ * @brief Put a 16-bit integer as big-endian to arbitrary location.
+ *
+ * Put a 16-bit integer, originally in host endianness, to a
+ * potentially unaligned memory location in big-endian format.
+ *
+ * @param val 16-bit integer in host endianness.
+ * @param dst Destination memory address to store the result.
+ */
+static inline void sys_put_be16(u16_t val, u8_t dst[2])
+{
+ dst[0] = val >> 8;
+ dst[1] = val;
+}
+
+/**
+ * @brief Put a 32-bit integer as big-endian to arbitrary location.
+ *
+ * Put a 32-bit integer, originally in host endianness, to a
+ * potentially unaligned memory location in big-endian format.
+ *
+ * @param val 32-bit integer in host endianness.
+ * @param dst Destination memory address to store the result.
+ */
+static inline void sys_put_be32(u32_t val, u8_t dst[4])
+{
+ sys_put_be16(val >> 16, dst);
+ sys_put_be16(val, &dst[2]);
+}
+
+/**
+ * @brief Put a 16-bit integer as little-endian to arbitrary location.
+ *
+ * Put a 16-bit integer, originally in host endianness, to a
+ * potentially unaligned memory location in little-endian format.
+ *
+ * @param val 16-bit integer in host endianness.
+ * @param dst Destination memory address to store the result.
+ */
+static inline void sys_put_le16(u16_t val, u8_t dst[2])
+{
+ dst[0] = val;
+ dst[1] = val >> 8;
+}
+
+/**
+ * @brief Put a 32-bit integer as little-endian to arbitrary location.
+ *
+ * Put a 32-bit integer, originally in host endianness, to a
+ * potentially unaligned memory location in little-endian format.
+ *
+ * @param val 32-bit integer in host endianness.
+ * @param dst Destination memory address to store the result.
+ */
+static inline void sys_put_le32(u32_t val, u8_t dst[4])
+{
+ sys_put_le16(val, dst);
+ sys_put_le16(val >> 16, &dst[2]);
+}
+
+/**
+ * @brief Put a 64-bit integer as little-endian to arbitrary location.
+ *
+ * Put a 64-bit integer, originally in host endianness, to a
+ * potentially unaligned memory location in little-endian format.
+ *
+ * @param val 64-bit integer in host endianness.
+ * @param dst Destination memory address to store the result.
+ */
+static inline void sys_put_le64(u64_t val, u8_t dst[8])
+{
+ sys_put_le32(val, dst);
+ sys_put_le32(val >> 32, &dst[4]);
+}
+
+/**
+ * @brief Get a 16-bit integer stored in big-endian format.
+ *
+ * Get a 16-bit integer, stored in big-endian format in a potentially
+ * unaligned memory location, and convert it to the host endianness.
+ *
+ * @param src Location of the big-endian 16-bit integer to get.
+ *
+ * @return 16-bit integer in host endianness.
+ */
+static inline u16_t sys_get_be16(const u8_t src[2])
+{
+ return ((u16_t)src[0] << 8) | src[1];
+}
+
+/**
+ * @brief Get a 32-bit integer stored in big-endian format.
+ *
+ * Get a 32-bit integer, stored in big-endian format in a potentially
+ * unaligned memory location, and convert it to the host endianness.
+ *
+ * @param src Location of the big-endian 32-bit integer to get.
+ *
+ * @return 32-bit integer in host endianness.
+ */
+static inline u32_t sys_get_be32(const u8_t src[4])
+{
+ return ((u32_t)sys_get_be16(&src[0]) << 16) | sys_get_be16(&src[2]);
+}
+
+/**
+ * @brief Get a 16-bit integer stored in little-endian format.
+ *
+ * Get a 16-bit integer, stored in little-endian format in a potentially
+ * unaligned memory location, and convert it to the host endianness.
+ *
+ * @param src Location of the little-endian 16-bit integer to get.
+ *
+ * @return 16-bit integer in host endianness.
+ */
+static inline u16_t sys_get_le16(const u8_t src[2])
+{
+ return ((u16_t)src[1] << 8) | src[0];
+}
+
+/**
+ * @brief Get a 32-bit integer stored in little-endian format.
+ *
+ * Get a 32-bit integer, stored in little-endian format in a potentially
+ * unaligned memory location, and convert it to the host endianness.
+ *
+ * @param src Location of the little-endian 32-bit integer to get.
+ *
+ * @return 32-bit integer in host endianness.
+ */
+static inline u32_t sys_get_le32(const u8_t src[4])
+{
+ return ((u32_t)sys_get_le16(&src[2]) << 16) | sys_get_le16(&src[0]);
+}
+
+/**
+ * @brief Get a 64-bit integer stored in little-endian format.
+ *
+ * Get a 64-bit integer, stored in little-endian format in a potentially
+ * unaligned memory location, and convert it to the host endianness.
+ *
+ * @param src Location of the little-endian 64-bit integer to get.
+ *
+ * @return 64-bit integer in host endianness.
+ */
+static inline u64_t sys_get_le64(const u8_t src[8])
+{
+ return ((u64_t)sys_get_le32(&src[4]) << 32) | sys_get_le32(&src[0]);
+}
+
+
+const char *bt_hex(const void *buf, size_t len);
+
+void mem_rcopy(u8_t *dst, u8_t const *src, u16_t len);
+
+void _set(void *to, uint8_t val, unsigned int len);
+
+unsigned int _copy(uint8_t *to, unsigned int to_len,
+ const uint8_t *from, unsigned int from_len);
+
+void _set(void *to, uint8_t val, unsigned int len);
+
+uint8_t _double_byte(uint8_t a);
+
+int _compare(const uint8_t *a, const uint8_t *b, size_t size);
+
+/**
+ * @brief Swap one buffer content into another
+ *
+ * Copy the content of src buffer into dst buffer in reversed order,
+ * i.e.: src[n] will be put in dst[end-n]
+ * Where n is an index and 'end' the last index in both arrays.
+ * The 2 memory pointers must be pointing to different areas, and have
+ * a minimum size of given length.
+ *
+ * @param dst A valid pointer on a memory area where to copy the data in
+ * @param src A valid pointer on a memory area where to copy the data from
+ * @param length Size of both dst and src memory areas
+ */
+static inline void sys_memcpy_swap(void *dst, const void *src, size_t length)
+{
+ __ASSERT(((src < dst && (src + length) <= dst) ||
+ (src > dst && (dst + length) <= src)),
+ "Source and destination buffers must not overlap");
+
+ src += length - 1;
+
+ for (; length > 0; length--) {
+ *((u8_t *)dst++) = *((u8_t *)src--);
+ }
+}
+
+/**
+ * @brief Swap buffer content
+ *
+ * In-place memory swap, where final content will be reversed.
+ * I.e.: buf[n] will be put in buf[end-n]
+ * Where n is an index and 'end' the last index of buf.
+ *
+ * @param buf A valid pointer on a memory area to swap
+ * @param length Size of buf memory area
+ */
+static inline void sys_mem_swap(void *buf, size_t length)
+{
+ size_t i;
+
+ for (i = 0; i < (length / 2); i++) {
+ u8_t tmp = ((u8_t *)buf)[i];
+
+ ((u8_t *)buf)[i] = ((u8_t *)buf)[length - 1 - i];
+ ((u8_t *)buf)[length - 1 - i] = tmp;
+ }
+}
+
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* _UTIL__H_ */
diff --git a/components/bt/ble_mesh/mesh_core/include/proxy.h b/components/bt/ble_mesh/mesh_core/include/proxy.h
new file mode 100644
index 0000000000..be4ba259b4
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/proxy.h
@@ -0,0 +1,37 @@
+/** @file
+ * @brief Bluetooth Mesh Proxy APIs.
+ */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __BT_MESH_PROXY_H
+#define __BT_MESH_PROXY_H
+
+#include
+/**
+ * @brief Bluetooth Mesh Proxy
+ * @defgroup bt_mesh_proxy Bluetooth Mesh Proxy
+ * @ingroup bt_mesh
+ * @{
+ */
+
+/**
+ * @brief Enable advertising with Node Identity.
+ *
+ * This API requires that GATT Proxy support has been enabled. Once called
+ * each subnet will start advertising using Node Identity for the next
+ * 60 seconds.
+ *
+ * @return 0 on success, or (negative) error code on failure.
+ */
+int bt_mesh_proxy_identity_enable(void);
+
+/**
+ * @}
+ */
+
+#endif /* __BT_MESH_PROXY_H */
diff --git a/components/bt/ble_mesh/mesh_core/include/settings.h b/components/bt/ble_mesh/mesh_core/include/settings.h
new file mode 100644
index 0000000000..a4377aa9c5
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/include/settings.h
@@ -0,0 +1,62 @@
+/*
+ * Copyright (c) 2018 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef _SETTINGS_H_
+#define _SETTINGS_H_
+
+#include "sdkconfig.h"
+
+#include "net.h"
+#include "mesh_access.h"
+#include "mesh_bearer_adapt.h"
+
+struct bt_settings_handler {
+ const char *name;
+ int (*set)(int argc, char **argv, char *val);
+ int (*commit)(void);
+ int (*export)(int (*func)(const char *name, char *val));
+};
+
+#define BT_SETTINGS_DEFINE(_name, _set, _commit, _export) \
+ const struct bt_settings_handler _name __aligned(4) \
+ __in_section(_bt_settings, static, _name) = { \
+ .name = STRINGIFY(_name), \
+ .set = _set, \
+ .commit = _commit, \
+ .export = _export, \
+ }
+
+/* Max settings key length (with all components) */
+#define BT_SETTINGS_KEY_MAX 36
+
+/* Base64-encoded string buffer size of in_size bytes */
+#define BT_SETTINGS_SIZE(in_size) ((((((in_size) - 1) / 3) * 4) + 4) + 1)
+
+/* Helpers for keys containing a bdaddr */
+void bt_settings_encode_key(char *path, size_t path_size, const char *subsys,
+ bt_addr_le_t *addr, const char *key);
+int bt_settings_decode_key(char *key, bt_addr_le_t *addr);
+
+void bt_mesh_store_net(void);
+void bt_mesh_store_iv(void);
+void bt_mesh_store_seq(void);
+void bt_mesh_store_rpl(struct bt_mesh_rpl *rpl);
+void bt_mesh_store_subnet(struct bt_mesh_subnet *sub);
+void bt_mesh_store_app_key(struct bt_mesh_app_key *key);
+void bt_mesh_store_hb_pub(void);
+void bt_mesh_store_cfg(void);
+void bt_mesh_store_mod_bind(struct bt_mesh_model *mod);
+void bt_mesh_store_mod_sub(struct bt_mesh_model *mod);
+void bt_mesh_store_mod_pub(struct bt_mesh_model *mod);
+
+void bt_mesh_clear_net(void);
+void bt_mesh_clear_subnet(struct bt_mesh_subnet *sub);
+void bt_mesh_clear_app_key(struct bt_mesh_app_key *key);
+void bt_mesh_clear_rpl(void);
+
+int bt_mesh_settings_init(void);
+
+#endif /* _SETTINGS_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/lpn.c b/components/bt/ble_mesh/mesh_core/lpn.c
new file mode 100644
index 0000000000..8c492135f7
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/lpn.c
@@ -0,0 +1,1054 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+#include "mesh_kernel.h"
+#ifdef CONFIG_BT_MESH_LOW_POWER
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_LOW_POWER)
+#include "mesh_trace.h"
+
+#include "crypto.h"
+#include "adv.h"
+#include "mesh.h"
+#include "net.h"
+#include "transport.h"
+#include "access.h"
+#include "beacon.h"
+#include "foundation.h"
+#include "lpn.h"
+
+#if defined(CONFIG_BT_MESH_LPN_AUTO)
+#define LPN_AUTO_TIMEOUT K_SECONDS(CONFIG_BT_MESH_LPN_AUTO_TIMEOUT)
+#else
+#define LPN_AUTO_TIMEOUT 0
+#endif
+
+#define LPN_RECV_DELAY CONFIG_BT_MESH_LPN_RECV_DELAY
+#define SCAN_LATENCY min(CONFIG_BT_MESH_LPN_SCAN_LATENCY, \
+ LPN_RECV_DELAY)
+
+#define FRIEND_REQ_RETRY_TIMEOUT K_SECONDS(CONFIG_BT_MESH_LPN_RETRY_TIMEOUT)
+
+#define FRIEND_REQ_WAIT K_MSEC(100)
+#define FRIEND_REQ_SCAN K_SECONDS(1)
+#define FRIEND_REQ_TIMEOUT (FRIEND_REQ_WAIT + FRIEND_REQ_SCAN)
+
+#define POLL_RETRY_TIMEOUT K_MSEC(100)
+
+#define REQ_RETRY_DURATION(lpn) (4 * (LPN_RECV_DELAY + (lpn)->adv_duration + \
+ (lpn)->recv_win + POLL_RETRY_TIMEOUT))
+
+#define POLL_TIMEOUT_INIT (CONFIG_BT_MESH_LPN_INIT_POLL_TIMEOUT * 100)
+#define POLL_TIMEOUT_MAX(lpn) ((CONFIG_BT_MESH_LPN_POLL_TIMEOUT * 100) - \
+ REQ_RETRY_DURATION(lpn))
+
+/* Update 4 to 20 for BQB test case MESH/NODE/FRND/LPM/BI-02-C */
+#define REQ_ATTEMPTS(lpn) (POLL_TIMEOUT_MAX(lpn) < K_SECONDS(3) ? 2 : 4)
+
+#define CLEAR_ATTEMPTS 2
+
+#define LPN_CRITERIA ((CONFIG_BT_MESH_LPN_MIN_QUEUE_SIZE) | \
+ (CONFIG_BT_MESH_LPN_RSSI_FACTOR << 3) | \
+ (CONFIG_BT_MESH_LPN_RECV_WIN_FACTOR << 5))
+
+#define POLL_TO(to) { (u8_t)((to) >> 16), (u8_t)((to) >> 8), (u8_t)(to) }
+#define LPN_POLL_TO POLL_TO(CONFIG_BT_MESH_LPN_POLL_TIMEOUT)
+
+/* 2 transmissions, 20ms interval */
+#define POLL_XMIT BT_MESH_TRANSMIT(1, 20)
+
+static void (*lpn_cb)(u16_t friend_addr, bool established);
+
+#if defined(CONFIG_BT_MESH_DEBUG_LOW_POWER)
+static const char *state2str(int state)
+{
+ switch (state) {
+ case BT_MESH_LPN_DISABLED:
+ return "disabled";
+ case BT_MESH_LPN_CLEAR:
+ return "clear";
+ case BT_MESH_LPN_TIMER:
+ return "timer";
+ case BT_MESH_LPN_ENABLED:
+ return "enabled";
+ case BT_MESH_LPN_REQ_WAIT:
+ return "req wait";
+ case BT_MESH_LPN_WAIT_OFFER:
+ return "wait offer";
+ case BT_MESH_LPN_ESTABLISHED:
+ return "established";
+ case BT_MESH_LPN_RECV_DELAY:
+ return "recv delay";
+ case BT_MESH_LPN_WAIT_UPDATE:
+ return "wait update";
+ default:
+ return "(unknown)";
+ }
+}
+#endif /* CONFIG_BT_MESH_DEBUG_LPN */
+
+static inline void lpn_set_state(int state)
+{
+#if defined(CONFIG_BT_MESH_DEBUG_LOW_POWER)
+ BT_DBG("%s -> %s", state2str(bt_mesh.lpn.state), state2str(state));
+#endif
+ bt_mesh.lpn.state = state;
+}
+
+static inline void group_zero(atomic_t *target)
+{
+#if CONFIG_BT_MESH_LPN_GROUPS > 32
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) {
+ atomic_set(&target[i], 0);
+ }
+#else
+ atomic_set(target, 0);
+#endif
+}
+
+static inline void group_set(atomic_t *target, atomic_t *source)
+{
+#if CONFIG_BT_MESH_LPN_GROUPS > 32
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) {
+ atomic_or(&target[i], atomic_get(&source[i]));
+ }
+#else
+ atomic_or(target, atomic_get(source));
+#endif
+}
+
+static inline void group_clear(atomic_t *target, atomic_t *source)
+{
+#if CONFIG_BT_MESH_LPN_GROUPS > 32
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) {
+ atomic_and(&target[i], ~atomic_get(&source[i]));
+ }
+#else
+ atomic_and(target, ~atomic_get(source));
+#endif
+}
+
+static void clear_friendship(bool force, bool disable);
+
+static void friend_clear_sent(int err, void *user_data)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+ /* We're switching away from Low Power behavior, so permanently
+ * enable scanning.
+ */
+ bt_mesh_scan_enable();
+
+ lpn->req_attempts++;
+
+ if (err) {
+ BT_ERR("Sending Friend Request failed (err %d)", err);
+ lpn_set_state(BT_MESH_LPN_ENABLED);
+ clear_friendship(false, lpn->disable);
+ return;
+ }
+
+ lpn_set_state(BT_MESH_LPN_CLEAR);
+ k_delayed_work_submit(&lpn->timer, FRIEND_REQ_TIMEOUT);
+}
+
+static const struct bt_mesh_send_cb clear_sent_cb = {
+ .end = friend_clear_sent,
+};
+
+static int send_friend_clear(void)
+{
+ struct bt_mesh_msg_ctx ctx = {
+ .net_idx = bt_mesh.sub[0].net_idx,
+ .app_idx = BT_MESH_KEY_UNUSED,
+ .addr = bt_mesh.lpn.frnd,
+ .send_ttl = 0,
+ };
+ struct bt_mesh_net_tx tx = {
+ .sub = &bt_mesh.sub[0],
+ .ctx = &ctx,
+ .src = bt_mesh_primary_addr(),
+ .xmit = bt_mesh_net_transmit_get(),
+ };
+ struct bt_mesh_ctl_friend_clear req = {
+ .lpn_addr = sys_cpu_to_be16(tx.src),
+ .lpn_counter = sys_cpu_to_be16(bt_mesh.lpn.counter),
+ };
+
+ BT_DBG("%s", __func__);
+
+ return bt_mesh_ctl_send(&tx, TRANS_CTL_OP_FRIEND_CLEAR, &req,
+ sizeof(req), NULL, &clear_sent_cb, NULL);
+}
+
+static void clear_friendship(bool force, bool disable)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+ BT_DBG("force %u disable %u", force, disable);
+
+ if (!force && lpn->established && !lpn->clear_success &&
+ lpn->req_attempts < CLEAR_ATTEMPTS) {
+ send_friend_clear();
+ lpn->disable = disable;
+ return;
+ }
+
+ bt_mesh_rx_reset();
+
+ k_delayed_work_cancel(&lpn->timer);
+
+ friend_cred_del(bt_mesh.sub[0].net_idx, lpn->frnd);
+
+ if (lpn->clear_success) {
+ lpn->old_friend = BT_MESH_ADDR_UNASSIGNED;
+ } else {
+ lpn->old_friend = lpn->frnd;
+ }
+
+ if (lpn_cb && lpn->frnd != BT_MESH_ADDR_UNASSIGNED) {
+ lpn_cb(lpn->frnd, false);
+ }
+
+ lpn->frnd = BT_MESH_ADDR_UNASSIGNED;
+ lpn->fsn = 0;
+ lpn->req_attempts = 0;
+ lpn->recv_win = 0;
+ lpn->queue_size = 0;
+ lpn->disable = 0;
+ lpn->sent_req = 0;
+ lpn->established = 0;
+ lpn->clear_success = 0;
+
+ group_zero(lpn->added);
+ group_zero(lpn->pending);
+ group_zero(lpn->to_remove);
+
+ /* Set this to 1 to force group subscription when the next
+ * Friendship is created, in case lpn->groups doesn't get
+ * modified meanwhile.
+ */
+ lpn->groups_changed = 1;
+
+ if (disable) {
+ lpn_set_state(BT_MESH_LPN_DISABLED);
+ return;
+ }
+
+ lpn_set_state(BT_MESH_LPN_ENABLED);
+ k_delayed_work_submit(&lpn->timer, FRIEND_REQ_RETRY_TIMEOUT);
+}
+
+static void friend_req_sent(u16_t duration, int err, void *user_data)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+ if (err) {
+ BT_ERR("Sending Friend Request failed (err %d)", err);
+ return;
+ }
+
+ lpn->adv_duration = duration;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) {
+ k_delayed_work_submit(&lpn->timer, FRIEND_REQ_WAIT);
+ lpn_set_state(BT_MESH_LPN_REQ_WAIT);
+ } else {
+ k_delayed_work_submit(&lpn->timer,
+ duration + FRIEND_REQ_TIMEOUT);
+ lpn_set_state(BT_MESH_LPN_WAIT_OFFER);
+ }
+}
+
+static const struct bt_mesh_send_cb friend_req_sent_cb = {
+ .start = friend_req_sent,
+};
+
+static int send_friend_req(struct bt_mesh_lpn *lpn)
+{
+ const struct bt_mesh_comp *comp = bt_mesh_comp_get();
+ struct bt_mesh_msg_ctx ctx = {
+ .net_idx = bt_mesh.sub[0].net_idx,
+ .app_idx = BT_MESH_KEY_UNUSED,
+ .addr = BT_MESH_ADDR_FRIENDS,
+ .send_ttl = 0,
+ };
+ struct bt_mesh_net_tx tx = {
+ .sub = &bt_mesh.sub[0],
+ .ctx = &ctx,
+ .src = bt_mesh_primary_addr(),
+ .xmit = POLL_XMIT,
+ };
+ struct bt_mesh_ctl_friend_req req = {
+ .criteria = LPN_CRITERIA,
+ .recv_delay = LPN_RECV_DELAY,
+ .poll_to = LPN_POLL_TO,
+ .prev_addr = lpn->old_friend,
+ .num_elem = comp->elem_count,
+ .lpn_counter = sys_cpu_to_be16(lpn->counter),
+ };
+
+ BT_DBG("%s", __func__);
+
+ return bt_mesh_ctl_send(&tx, TRANS_CTL_OP_FRIEND_REQ, &req,
+ sizeof(req), NULL, &friend_req_sent_cb, NULL);
+}
+
+static void req_sent(u16_t duration, int err, void *user_data)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+#if defined(CONFIG_BT_MESH_DEBUG_LOW_POWER)
+ BT_DBG("req 0x%02x duration %u err %d state %s",
+ lpn->sent_req, duration, err, state2str(lpn->state));
+#endif
+
+ if (err) {
+ BT_ERR("Sending request failed (err %d)", err);
+ lpn->sent_req = 0;
+ group_zero(lpn->pending);
+ return;
+ }
+
+ lpn->req_attempts++;
+ lpn->adv_duration = duration;
+
+ if (lpn->established || IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) {
+ lpn_set_state(BT_MESH_LPN_RECV_DELAY);
+ /* We start scanning a bit early to elimitate risk of missing
+ * response data due to HCI and other latencies.
+ */
+ k_delayed_work_submit(&lpn->timer,
+ LPN_RECV_DELAY - SCAN_LATENCY);
+ } else {
+ lpn_set_state(BT_MESH_LPN_OFFER_RECV);
+ k_delayed_work_submit(&lpn->timer,
+ LPN_RECV_DELAY + duration +
+ lpn->recv_win);
+ }
+}
+
+static const struct bt_mesh_send_cb req_sent_cb = {
+ .start = req_sent,
+};
+
+static int send_friend_poll(void)
+{
+ struct bt_mesh_msg_ctx ctx = {
+ .net_idx = bt_mesh.sub[0].net_idx,
+ .app_idx = BT_MESH_KEY_UNUSED,
+ .addr = bt_mesh.lpn.frnd,
+ .send_ttl = 0,
+ };
+ struct bt_mesh_net_tx tx = {
+ .sub = &bt_mesh.sub[0],
+ .ctx = &ctx,
+ .src = bt_mesh_primary_addr(),
+ .xmit = POLL_XMIT,
+ .friend_cred = true,
+ };
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+ u8_t fsn = lpn->fsn;
+ int err;
+
+ BT_DBG("lpn->sent_req 0x%02x", lpn->sent_req);
+
+ if (lpn->sent_req) {
+ if (lpn->sent_req != TRANS_CTL_OP_FRIEND_POLL) {
+ lpn->pending_poll = 1;
+ }
+
+ return 0;
+ }
+
+ err = bt_mesh_ctl_send(&tx, TRANS_CTL_OP_FRIEND_POLL, &fsn, 1,
+ NULL, &req_sent_cb, NULL);
+ if (err == 0) {
+ lpn->pending_poll = 0;
+ lpn->sent_req = TRANS_CTL_OP_FRIEND_POLL;
+ }
+
+ return err;
+}
+
+void bt_mesh_lpn_disable(bool force)
+{
+ if (bt_mesh.lpn.state == BT_MESH_LPN_DISABLED) {
+ return;
+ }
+
+ clear_friendship(force, true);
+}
+
+int bt_mesh_lpn_set(bool enable)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+ if (enable) {
+ if (lpn->state != BT_MESH_LPN_DISABLED) {
+ return 0;
+ }
+ } else {
+ if (lpn->state == BT_MESH_LPN_DISABLED) {
+ return 0;
+ }
+ }
+
+ if (!bt_mesh_is_provisioned()) {
+ if (enable) {
+ lpn_set_state(BT_MESH_LPN_ENABLED);
+ } else {
+ lpn_set_state(BT_MESH_LPN_DISABLED);
+ }
+
+ return 0;
+ }
+
+ if (enable) {
+ lpn_set_state(BT_MESH_LPN_ENABLED);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) {
+ bt_mesh_scan_disable();
+ }
+
+ send_friend_req(lpn);
+ } else {
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_AUTO) &&
+ lpn->state == BT_MESH_LPN_TIMER) {
+ k_delayed_work_cancel(&lpn->timer);
+ lpn_set_state(BT_MESH_LPN_DISABLED);
+ } else {
+ bt_mesh_lpn_disable(false);
+ }
+ }
+
+ return 0;
+}
+
+static void friend_response_received(struct bt_mesh_lpn *lpn)
+{
+ BT_DBG("lpn->sent_req 0x%02x", lpn->sent_req);
+
+ if (lpn->sent_req == TRANS_CTL_OP_FRIEND_POLL) {
+ lpn->fsn++;
+ }
+
+ k_delayed_work_cancel(&lpn->timer);
+ bt_mesh_scan_disable();
+ lpn_set_state(BT_MESH_LPN_ESTABLISHED);
+ lpn->req_attempts = 0;
+ lpn->sent_req = 0;
+}
+
+void bt_mesh_lpn_msg_received(struct bt_mesh_net_rx *rx)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+ if (lpn->state == BT_MESH_LPN_TIMER) {
+ BT_DBG("Restarting establishment timer");
+ k_delayed_work_submit(&lpn->timer, LPN_AUTO_TIMEOUT);
+ return;
+ }
+
+ if (lpn->sent_req != TRANS_CTL_OP_FRIEND_POLL) {
+ BT_WARN("Unexpected message withouth a preceding Poll");
+ return;
+ }
+
+ friend_response_received(lpn);
+
+ BT_DBG("Requesting more messages from Friend");
+
+ send_friend_poll();
+}
+
+int bt_mesh_lpn_friend_offer(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_offer *msg = (void *)buf->data;
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+ struct bt_mesh_subnet *sub = rx->sub;
+ struct friend_cred *cred;
+ u16_t frnd_counter;
+ int err;
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Offer");
+ return -EINVAL;
+ }
+
+ if (lpn->state != BT_MESH_LPN_WAIT_OFFER) {
+ BT_WARN("Ignoring unexpected Friend Offer");
+ return 0;
+ }
+
+ if (!msg->recv_win) {
+ BT_WARN("Prohibited ReceiveWindow value");
+ return -EINVAL;
+ }
+
+ frnd_counter = sys_be16_to_cpu(msg->frnd_counter);
+
+ BT_DBG("recv_win %u queue_size %u sub_list_size %u rssi %d counter %u",
+ msg->recv_win, msg->queue_size, msg->sub_list_size, msg->rssi,
+ frnd_counter);
+
+ lpn->frnd = rx->ctx.addr;
+
+ cred = friend_cred_create(sub, lpn->frnd, lpn->counter, frnd_counter);
+ if (!cred) {
+ lpn->frnd = BT_MESH_ADDR_UNASSIGNED;
+ return -ENOMEM;
+ }
+
+ /* TODO: Add offer acceptance criteria check */
+
+ k_delayed_work_cancel(&lpn->timer);
+
+ lpn->recv_win = msg->recv_win;
+ lpn->queue_size = msg->queue_size;
+
+ err = send_friend_poll();
+ if (err) {
+ friend_cred_clear(cred);
+ lpn->frnd = BT_MESH_ADDR_UNASSIGNED;
+ lpn->recv_win = 0;
+ lpn->queue_size = 0;
+ return err;
+ }
+
+ lpn->counter++;
+
+ return 0;
+}
+
+int bt_mesh_lpn_friend_clear_cfm(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_clear_confirm *msg = (void *)buf->data;
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+ u16_t addr, counter;
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Clear Confirm");
+ return -EINVAL;
+ }
+
+ if (lpn->state != BT_MESH_LPN_CLEAR) {
+ BT_WARN("Ignoring unexpected Friend Clear Confirm");
+ return 0;
+ }
+
+ addr = sys_be16_to_cpu(msg->lpn_addr);
+ counter = sys_be16_to_cpu(msg->lpn_counter);
+
+ BT_DBG("LPNAddress 0x%04x LPNCounter 0x%04x", addr, counter);
+
+ if (addr != bt_mesh_primary_addr() || counter != lpn->counter) {
+ BT_WARN("Invalid parameters in Friend Clear Confirm");
+ return 0;
+ }
+
+ lpn->clear_success = 1;
+ clear_friendship(false, lpn->disable);
+
+ return 0;
+}
+
+static void lpn_group_add(u16_t group)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+ u16_t *free_slot = NULL;
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(lpn->groups); i++) {
+ if (lpn->groups[i] == group) {
+ atomic_clear_bit(lpn->to_remove, i);
+ return;
+ }
+
+ if (!free_slot && lpn->groups[i] == BT_MESH_ADDR_UNASSIGNED) {
+ free_slot = &lpn->groups[i];
+ }
+ }
+
+ if (!free_slot) {
+ BT_WARN("Friend Subscription List exceeded!");
+ return;
+ }
+
+ *free_slot = group;
+ lpn->groups_changed = 1;
+}
+
+static void lpn_group_del(u16_t group)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(lpn->groups); i++) {
+ if (lpn->groups[i] == group) {
+ if (atomic_test_bit(lpn->added, i) ||
+ atomic_test_bit(lpn->pending, i)) {
+ atomic_set_bit(lpn->to_remove, i);
+ lpn->groups_changed = 1;
+ } else {
+ lpn->groups[i] = BT_MESH_ADDR_UNASSIGNED;
+ }
+ }
+ }
+}
+
+static inline int group_popcount(atomic_t *target)
+{
+#if CONFIG_BT_MESH_LPN_GROUPS > 32
+ int i, count = 0;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.lpn.added); i++) {
+ count += popcount(atomic_get(&target[i]));
+ }
+#else
+ return popcount(atomic_get(target));
+#endif
+}
+
+static bool sub_update(u8_t op)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+ int added_count = group_popcount(lpn->added);
+ struct bt_mesh_msg_ctx ctx = {
+ .net_idx = bt_mesh.sub[0].net_idx,
+ .app_idx = BT_MESH_KEY_UNUSED,
+ .addr = lpn->frnd,
+ .send_ttl = 0,
+ };
+ struct bt_mesh_net_tx tx = {
+ .sub = &bt_mesh.sub[0],
+ .ctx = &ctx,
+ .src = bt_mesh_primary_addr(),
+ .xmit = POLL_XMIT,
+ .friend_cred = true,
+ };
+ struct bt_mesh_ctl_friend_sub req;
+ size_t i, g;
+
+ BT_DBG("op 0x%02x sent_req 0x%02x", op, lpn->sent_req);
+
+ if (lpn->sent_req) {
+ return false;
+ }
+
+ for (i = 0, g = 0; i < ARRAY_SIZE(lpn->groups); i++) {
+ if (lpn->groups[i] == BT_MESH_ADDR_UNASSIGNED) {
+ continue;
+ }
+
+ if (op == TRANS_CTL_OP_FRIEND_SUB_ADD) {
+ if (atomic_test_bit(lpn->added, i)) {
+ continue;
+ }
+ } else {
+ if (!atomic_test_bit(lpn->to_remove, i)) {
+ continue;
+ }
+ }
+
+ if (added_count + g >= lpn->queue_size) {
+ BT_WARN("Friend Queue Size exceeded");
+ break;
+ }
+
+ req.addr_list[g++] = sys_cpu_to_be16(lpn->groups[i]);
+ atomic_set_bit(lpn->pending, i);
+
+ if (g == ARRAY_SIZE(req.addr_list)) {
+ break;
+ }
+ }
+
+ if (g == 0) {
+ group_zero(lpn->pending);
+ return false;
+ }
+
+ req.xact = lpn->xact_next++;
+
+ if (bt_mesh_ctl_send(&tx, op, &req, 1 + g * 2, NULL,
+ &req_sent_cb, NULL) < 0) {
+ group_zero(lpn->pending);
+ return false;
+ }
+
+ lpn->xact_pending = req.xact;
+ lpn->sent_req = op;
+ return true;
+}
+
+static void update_timeout(struct bt_mesh_lpn *lpn)
+{
+ if (lpn->established) {
+ BT_WARN("No response from Friend during ReceiveWindow");
+ bt_mesh_scan_disable();
+ lpn_set_state(BT_MESH_LPN_ESTABLISHED);
+ k_delayed_work_submit(&lpn->timer, POLL_RETRY_TIMEOUT);
+ } else {
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) {
+ bt_mesh_scan_disable();
+ }
+
+ if (lpn->req_attempts < 6) {
+ BT_WARN("Retrying first Friend Poll");
+ lpn->sent_req = 0;
+ if (send_friend_poll() == 0) {
+ return;
+ }
+ }
+
+ BT_ERR("Timed out waiting for first Friend Update");
+ clear_friendship(false, false);
+ }
+}
+
+static void lpn_timeout(struct k_work *work)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+#if defined(CONFIG_BT_MESH_DEBUG_LOW_POWER)
+ BT_DBG("state: %s", state2str(lpn->state));
+#endif
+
+ switch (lpn->state) {
+ case BT_MESH_LPN_DISABLED:
+ break;
+ case BT_MESH_LPN_CLEAR:
+ clear_friendship(false, bt_mesh.lpn.disable);
+ break;
+ case BT_MESH_LPN_TIMER:
+ BT_DBG("Starting to look for Friend nodes");
+ lpn_set_state(BT_MESH_LPN_ENABLED);
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) {
+ bt_mesh_scan_disable();
+ }
+ /* fall through */
+ case BT_MESH_LPN_ENABLED:
+ send_friend_req(lpn);
+ break;
+ case BT_MESH_LPN_REQ_WAIT:
+ bt_mesh_scan_enable();
+ k_delayed_work_submit(&lpn->timer,
+ lpn->adv_duration + FRIEND_REQ_SCAN);
+ lpn_set_state(BT_MESH_LPN_WAIT_OFFER);
+ break;
+ case BT_MESH_LPN_WAIT_OFFER:
+ BT_WARN("No acceptable Friend Offers received");
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) {
+ bt_mesh_scan_disable();
+ }
+ lpn->counter++;
+ lpn_set_state(BT_MESH_LPN_ENABLED);
+ k_delayed_work_submit(&lpn->timer, FRIEND_REQ_RETRY_TIMEOUT);
+ break;
+ case BT_MESH_LPN_OFFER_RECV:
+ BT_WARN("No Friend Update received after the first Friend Poll");
+ lpn->sent_req = 0;
+ send_friend_poll();
+ break;
+ case BT_MESH_LPN_ESTABLISHED:
+ if (lpn->req_attempts < REQ_ATTEMPTS(lpn)) {
+ u8_t req = lpn->sent_req;
+
+ lpn->sent_req = 0;
+
+ if (!req || req == TRANS_CTL_OP_FRIEND_POLL) {
+ send_friend_poll();
+ } else {
+ sub_update(req);
+ }
+
+ break;
+ }
+
+ BT_ERR("No response from Friend after %u retries",
+ lpn->req_attempts);
+ lpn->req_attempts = 0;
+ clear_friendship(false, false);
+ break;
+ case BT_MESH_LPN_RECV_DELAY:
+ k_delayed_work_submit(&lpn->timer,
+ lpn->adv_duration + SCAN_LATENCY +
+ lpn->recv_win);
+ bt_mesh_scan_enable();
+ lpn_set_state(BT_MESH_LPN_WAIT_UPDATE);
+ break;
+ case BT_MESH_LPN_WAIT_UPDATE:
+ update_timeout(lpn);
+ break;
+ default:
+ __ASSERT(0, "Unhandled LPN state");
+ break;
+ }
+}
+
+void bt_mesh_lpn_group_add(u16_t group)
+{
+ BT_DBG("group 0x%04x", group);
+
+ lpn_group_add(group);
+
+ if (!bt_mesh_lpn_established() || bt_mesh.lpn.sent_req) {
+ return;
+ }
+
+ sub_update(TRANS_CTL_OP_FRIEND_SUB_ADD);
+}
+
+void bt_mesh_lpn_group_del(u16_t *groups, size_t group_count)
+{
+ int i;
+
+ for (i = 0; i < group_count; i++) {
+ if (groups[i] != BT_MESH_ADDR_UNASSIGNED) {
+ BT_DBG("group 0x%04x", groups[i]);
+ lpn_group_del(groups[i]);
+ }
+ }
+
+ if (!bt_mesh_lpn_established() || bt_mesh.lpn.sent_req) {
+ return;
+ }
+
+ sub_update(TRANS_CTL_OP_FRIEND_SUB_REM);
+}
+
+static s32_t poll_timeout(struct bt_mesh_lpn *lpn)
+{
+ /* If we're waiting for segment acks keep polling at high freq */
+ if (bt_mesh_tx_in_progress()) {
+ return min(POLL_TIMEOUT_MAX(lpn), K_SECONDS(1));
+ }
+
+ if (lpn->poll_timeout < POLL_TIMEOUT_MAX(lpn)) {
+ lpn->poll_timeout *= 2;
+ lpn->poll_timeout = min(lpn->poll_timeout,
+ POLL_TIMEOUT_MAX(lpn));
+ }
+
+ BT_DBG("Poll Timeout is %ums", lpn->poll_timeout);
+
+ return lpn->poll_timeout;
+}
+
+int bt_mesh_lpn_friend_sub_cfm(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_sub_confirm *msg = (void *)buf->data;
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Subscription Confirm");
+ return -EINVAL;
+ }
+
+ BT_DBG("xact 0x%02x", msg->xact);
+
+ if (!lpn->sent_req) {
+ BT_WARN("No pending subscription list message");
+ return 0;
+ }
+
+ if (msg->xact != lpn->xact_pending) {
+ BT_WARN("Transaction mismatch (0x%02x != 0x%02x)",
+ msg->xact, lpn->xact_pending);
+ return 0;
+ }
+
+ if (lpn->sent_req == TRANS_CTL_OP_FRIEND_SUB_ADD) {
+ group_set(lpn->added, lpn->pending);
+ group_zero(lpn->pending);
+ } else if (lpn->sent_req == TRANS_CTL_OP_FRIEND_SUB_REM) {
+ int i;
+
+ group_clear(lpn->added, lpn->pending);
+
+ for (i = 0; i < ARRAY_SIZE(lpn->groups); i++) {
+ if (atomic_test_and_clear_bit(lpn->pending, i) &&
+ atomic_test_and_clear_bit(lpn->to_remove, i)) {
+ lpn->groups[i] = BT_MESH_ADDR_UNASSIGNED;
+ }
+ }
+ } else {
+ BT_WARN("Unexpected Friend Subscription Confirm");
+ return 0;
+ }
+
+ friend_response_received(lpn);
+
+ if (lpn->groups_changed) {
+ sub_update(TRANS_CTL_OP_FRIEND_SUB_ADD);
+ sub_update(TRANS_CTL_OP_FRIEND_SUB_REM);
+
+ if (!lpn->sent_req) {
+ lpn->groups_changed = 0;
+ }
+ }
+
+ if (lpn->pending_poll) {
+ send_friend_poll();
+ }
+
+ if (!lpn->sent_req) {
+ k_delayed_work_submit(&lpn->timer, poll_timeout(lpn));
+ }
+
+ return 0;
+}
+
+int bt_mesh_lpn_friend_update(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_ctl_friend_update *msg = (void *)buf->data;
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+ struct bt_mesh_subnet *sub = rx->sub;
+ u32_t iv_index;
+
+ if (buf->len < sizeof(*msg)) {
+ BT_WARN("Too short Friend Update");
+ return -EINVAL;
+ }
+
+ if (lpn->sent_req != TRANS_CTL_OP_FRIEND_POLL) {
+ BT_WARN("Unexpected friend update");
+ return 0;
+ }
+
+ if (sub->kr_phase == BT_MESH_KR_PHASE_2 && !rx->new_key) {
+ BT_WARN("Ignoring Phase 2 KR Update secured using old key");
+ return 0;
+ }
+
+ if (bt_mesh.ivu_initiator &&
+ bt_mesh.iv_update == BT_MESH_IV_UPDATE(msg->flags)) {
+ bt_mesh_beacon_ivu_initiator(false);
+ }
+
+ if (!lpn->established) {
+ /* This is normally checked on the transport layer, however
+ * in this state we're also still accepting master
+ * credentials so we need to ensure the right ones (Friend
+ * Credentials) were used for this message.
+ */
+ if (!rx->friend_cred) {
+ BT_WARN("Friend Update with wrong credentials");
+ return -EINVAL;
+ }
+
+ lpn->established = 1;
+
+ BT_INFO("Friendship established with 0x%04x", lpn->frnd);
+
+ if (lpn_cb) {
+ lpn_cb(lpn->frnd, true);
+ }
+
+ /* Set initial poll timeout */
+ lpn->poll_timeout = min(POLL_TIMEOUT_MAX(lpn),
+ POLL_TIMEOUT_INIT);
+ }
+
+ friend_response_received(lpn);
+
+ iv_index = sys_be32_to_cpu(msg->iv_index);
+
+ BT_DBG("flags 0x%02x iv_index 0x%08x md %u", msg->flags, iv_index,
+ msg->md);
+
+ if (bt_mesh_kr_update(sub, BT_MESH_KEY_REFRESH(msg->flags),
+ rx->new_key)) {
+ bt_mesh_net_beacon_update(sub);
+ }
+
+ bt_mesh_net_iv_update(iv_index, BT_MESH_IV_UPDATE(msg->flags));
+
+ if (lpn->groups_changed) {
+ sub_update(TRANS_CTL_OP_FRIEND_SUB_ADD);
+ sub_update(TRANS_CTL_OP_FRIEND_SUB_REM);
+
+ if (!lpn->sent_req) {
+ lpn->groups_changed = 0;
+ }
+ }
+
+ if (msg->md) {
+ BT_DBG("Requesting for more messages");
+ send_friend_poll();
+ }
+
+ if (!lpn->sent_req) {
+ k_delayed_work_submit(&lpn->timer, poll_timeout(lpn));
+ }
+
+ return 0;
+}
+
+int bt_mesh_lpn_poll(void)
+{
+ if (!bt_mesh.lpn.established) {
+ return -EAGAIN;
+ }
+
+ BT_DBG("Requesting more messages");
+
+ return send_friend_poll();
+}
+
+void bt_mesh_lpn_set_cb(void (*cb)(u16_t friend_addr, bool established))
+{
+ lpn_cb = cb;
+}
+
+int bt_mesh_lpn_init(void)
+{
+ struct bt_mesh_lpn *lpn = &bt_mesh.lpn;
+
+ BT_DBG("%s", __func__);
+
+ k_delayed_work_init(&lpn->timer, lpn_timeout);
+
+ if (lpn->state == BT_MESH_LPN_ENABLED) {
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_ESTABLISHMENT)) {
+ bt_mesh_scan_disable();
+ } else {
+ bt_mesh_scan_enable();
+ }
+
+ send_friend_req(lpn);
+ } else {
+ bt_mesh_scan_enable();
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LPN_AUTO)) {
+ BT_DBG("Waiting %u ms for messages", LPN_AUTO_TIMEOUT);
+ lpn_set_state(BT_MESH_LPN_TIMER);
+ k_delayed_work_submit(&lpn->timer, LPN_AUTO_TIMEOUT);
+ }
+ }
+
+ return 0;
+}
+#endif /* CONFIG_BT_MESH_LOW_POWER */
+#endif /* #if CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/lpn.h b/components/bt/ble_mesh/mesh_core/lpn.h
new file mode 100644
index 0000000000..6dc6dd960f
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/lpn.h
@@ -0,0 +1,62 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+int bt_mesh_lpn_friend_update(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf);
+int bt_mesh_lpn_friend_offer(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf);
+int bt_mesh_lpn_friend_clear_cfm(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf);
+int bt_mesh_lpn_friend_sub_cfm(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf);
+
+static inline bool bt_mesh_lpn_established(void)
+{
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+ return bt_mesh.lpn.established;
+#else
+ return false;
+#endif
+}
+
+static inline bool bt_mesh_lpn_match(u16_t addr)
+{
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+ if (bt_mesh_lpn_established()) {
+ return (addr == bt_mesh.lpn.frnd);
+ }
+#endif
+ return false;
+}
+
+static inline bool bt_mesh_lpn_waiting_update(void)
+{
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+ return (bt_mesh.lpn.state == BT_MESH_LPN_WAIT_UPDATE);
+#else
+ return false;
+#endif
+}
+
+static inline bool bt_mesh_lpn_timer(void)
+{
+#if defined(CONFIG_BT_MESH_LPN_AUTO)
+ return (bt_mesh.lpn.state == BT_MESH_LPN_TIMER);
+#else
+ return false;
+#endif
+}
+
+void bt_mesh_lpn_msg_received(struct bt_mesh_net_rx *rx);
+
+void bt_mesh_lpn_group_add(u16_t group);
+void bt_mesh_lpn_group_del(u16_t *groups, size_t group_count);
+
+void bt_mesh_lpn_disable(bool force);
+
+int bt_mesh_lpn_init(void);
diff --git a/components/bt/ble_mesh/mesh_core/mesh.h b/components/bt/ble_mesh/mesh_core/mesh.h
new file mode 100644
index 0000000000..b8ca726566
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh.h
@@ -0,0 +1,19 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#define BT_MESH_KEY_PRIMARY 0x0000
+#define BT_MESH_KEY_ANY 0xffff
+
+#define BT_MESH_ADDR_IS_UNICAST(addr) ((addr) && (addr) < 0x8000)
+#define BT_MESH_ADDR_IS_GROUP(addr) ((addr) >= 0xc000 && (addr) <= 0xff00)
+#define BT_MESH_ADDR_IS_VIRTUAL(addr) ((addr) >= 0x8000 && (addr) < 0xc000)
+#define BT_MESH_ADDR_IS_RFU(addr) ((addr) >= 0xff00 && (addr) <= 0xfffb)
+
+struct bt_mesh_net;
+
+bool bt_mesh_is_provisioned(void);
diff --git a/components/bt/ble_mesh/mesh_core/mesh_aes_encrypt.c b/components/bt/ble_mesh/mesh_core/mesh_aes_encrypt.c
new file mode 100644
index 0000000000..db1d8b8489
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_aes_encrypt.c
@@ -0,0 +1,415 @@
+/* aes_encrypt.c - TinyCrypt implementation of AES encryption procedure */
+
+/*
+ * Copyright (C) 2017 by Intel Corporation, All Rights Reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * - Redistributions of source code must retain the above copyright notice,
+ * this list of conditions and the following disclaimer.
+ *
+ * - Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * - Neither the name of Intel Corporation nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
+ * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
+ * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
+ * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
+ * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
+ * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
+ * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include "mesh_aes_encrypt.h"
+#include "mesh_util.h"
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+
+/* max number of calls until change the key (2^48).*/
+const static uint64_t MAX_CALLS = ((uint64_t)1 << 48);
+
+/*
+ * gf_wrap -- In our implementation, GF(2^128) is represented as a 16 byte
+ * array with byte 0 the most significant and byte 15 the least significant.
+ * High bit carry reduction is based on the primitive polynomial
+ *
+ * X^128 + X^7 + X^2 + X + 1,
+ *
+ * which leads to the reduction formula X^128 = X^7 + X^2 + X + 1. Indeed,
+ * since 0 = (X^128 + X^7 + X^2 + 1) mod (X^128 + X^7 + X^2 + X + 1) and since
+ * addition of polynomials with coefficients in Z/Z(2) is just XOR, we can
+ * add X^128 to both sides to get
+ *
+ * X^128 = (X^7 + X^2 + X + 1) mod (X^128 + X^7 + X^2 + X + 1)
+ *
+ * and the coefficients of the polynomial on the right hand side form the
+ * string 1000 0111 = 0x87, which is the value of gf_wrap.
+ *
+ * This gets used in the following way. Doubling in GF(2^128) is just a left
+ * shift by 1 bit, except when the most significant bit is 1. In the latter
+ * case, the relation X^128 = X^7 + X^2 + X + 1 says that the high order bit
+ * that overflows beyond 128 bits can be replaced by addition of
+ * X^7 + X^2 + X + 1 <--> 0x87 to the low order 128 bits. Since addition
+ * in GF(2^128) is represented by XOR, we therefore only have to XOR 0x87
+ * into the low order byte after a left shift when the starting high order
+ * bit is 1.
+ */
+const unsigned char gf_wrap = 0x87;
+
+
+static const uint8_t sbox[256] = {
+ 0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, 0x30, 0x01, 0x67, 0x2b,
+ 0xfe, 0xd7, 0xab, 0x76, 0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59, 0x47, 0xf0,
+ 0xad, 0xd4, 0xa2, 0xaf, 0x9c, 0xa4, 0x72, 0xc0, 0xb7, 0xfd, 0x93, 0x26,
+ 0x36, 0x3f, 0xf7, 0xcc, 0x34, 0xa5, 0xe5, 0xf1, 0x71, 0xd8, 0x31, 0x15,
+ 0x04, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x05, 0x9a, 0x07, 0x12, 0x80, 0xe2,
+ 0xeb, 0x27, 0xb2, 0x75, 0x09, 0x83, 0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0,
+ 0x52, 0x3b, 0xd6, 0xb3, 0x29, 0xe3, 0x2f, 0x84, 0x53, 0xd1, 0x00, 0xed,
+ 0x20, 0xfc, 0xb1, 0x5b, 0x6a, 0xcb, 0xbe, 0x39, 0x4a, 0x4c, 0x58, 0xcf,
+ 0xd0, 0xef, 0xaa, 0xfb, 0x43, 0x4d, 0x33, 0x85, 0x45, 0xf9, 0x02, 0x7f,
+ 0x50, 0x3c, 0x9f, 0xa8, 0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5,
+ 0xbc, 0xb6, 0xda, 0x21, 0x10, 0xff, 0xf3, 0xd2, 0xcd, 0x0c, 0x13, 0xec,
+ 0x5f, 0x97, 0x44, 0x17, 0xc4, 0xa7, 0x7e, 0x3d, 0x64, 0x5d, 0x19, 0x73,
+ 0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, 0x46, 0xee, 0xb8, 0x14,
+ 0xde, 0x5e, 0x0b, 0xdb, 0xe0, 0x32, 0x3a, 0x0a, 0x49, 0x06, 0x24, 0x5c,
+ 0xc2, 0xd3, 0xac, 0x62, 0x91, 0x95, 0xe4, 0x79, 0xe7, 0xc8, 0x37, 0x6d,
+ 0x8d, 0xd5, 0x4e, 0xa9, 0x6c, 0x56, 0xf4, 0xea, 0x65, 0x7a, 0xae, 0x08,
+ 0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6, 0xb4, 0xc6, 0xe8, 0xdd, 0x74, 0x1f,
+ 0x4b, 0xbd, 0x8b, 0x8a, 0x70, 0x3e, 0xb5, 0x66, 0x48, 0x03, 0xf6, 0x0e,
+ 0x61, 0x35, 0x57, 0xb9, 0x86, 0xc1, 0x1d, 0x9e, 0xe1, 0xf8, 0x98, 0x11,
+ 0x69, 0xd9, 0x8e, 0x94, 0x9b, 0x1e, 0x87, 0xe9, 0xce, 0x55, 0x28, 0xdf,
+ 0x8c, 0xa1, 0x89, 0x0d, 0xbf, 0xe6, 0x42, 0x68, 0x41, 0x99, 0x2d, 0x0f,
+ 0xb0, 0x54, 0xbb, 0x16
+};
+
+static inline unsigned int rotword(unsigned int a)
+{
+ return (((a) >> 24) | ((a) << 8));
+}
+
+#define subbyte(a, o)(sbox[((a) >> (o))&0xff] << (o))
+#define subword(a)(subbyte(a, 24)|subbyte(a, 16)|subbyte(a, 8)|subbyte(a, 0))
+
+int tc_aes128_set_encrypt_key(TCAesKeySched_t s, const uint8_t *k)
+{
+ const unsigned int rconst[11] = {
+ 0x00000000, 0x01000000, 0x02000000, 0x04000000, 0x08000000, 0x10000000,
+ 0x20000000, 0x40000000, 0x80000000, 0x1b000000, 0x36000000
+ };
+ unsigned int i;
+ unsigned int t;
+
+ if (s == (TCAesKeySched_t) 0) {
+ return TC_CRYPTO_FAIL;
+ } else if (k == (const uint8_t *) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ for (i = 0; i < Nk; ++i) {
+ s->words[i] = (k[Nb * i] << 24) | (k[Nb * i + 1] << 16) |
+ (k[Nb * i + 2] << 8) | (k[Nb * i + 3]);
+ }
+
+ for (; i < (Nb * (Nr + 1)); ++i) {
+ t = s->words[i - 1];
+ if ((i % Nk) == 0) {
+ t = subword(rotword(t)) ^ rconst[i / Nk];
+ }
+ s->words[i] = s->words[i - Nk] ^ t;
+ }
+
+ return TC_CRYPTO_SUCCESS;
+}
+
+static inline void add_round_key(uint8_t *s, const unsigned int *k)
+{
+ s[0] ^= (uint8_t)(k[0] >> 24); s[1] ^= (uint8_t)(k[0] >> 16);
+ s[2] ^= (uint8_t)(k[0] >> 8); s[3] ^= (uint8_t)(k[0]);
+ s[4] ^= (uint8_t)(k[1] >> 24); s[5] ^= (uint8_t)(k[1] >> 16);
+ s[6] ^= (uint8_t)(k[1] >> 8); s[7] ^= (uint8_t)(k[1]);
+ s[8] ^= (uint8_t)(k[2] >> 24); s[9] ^= (uint8_t)(k[2] >> 16);
+ s[10] ^= (uint8_t)(k[2] >> 8); s[11] ^= (uint8_t)(k[2]);
+ s[12] ^= (uint8_t)(k[3] >> 24); s[13] ^= (uint8_t)(k[3] >> 16);
+ s[14] ^= (uint8_t)(k[3] >> 8); s[15] ^= (uint8_t)(k[3]);
+}
+
+static inline void sub_bytes(uint8_t *s)
+{
+ unsigned int i;
+
+ for (i = 0; i < (Nb * Nk); ++i) {
+ s[i] = sbox[s[i]];
+ }
+}
+
+#define triple(a)(_double_byte(a)^(a))
+
+static inline void mult_row_column(uint8_t *out, const uint8_t *in)
+{
+ out[0] = _double_byte(in[0]) ^ triple(in[1]) ^ in[2] ^ in[3];
+ out[1] = in[0] ^ _double_byte(in[1]) ^ triple(in[2]) ^ in[3];
+ out[2] = in[0] ^ in[1] ^ _double_byte(in[2]) ^ triple(in[3]);
+ out[3] = triple(in[0]) ^ in[1] ^ in[2] ^ _double_byte(in[3]);
+}
+
+static inline void mix_columns(uint8_t *s)
+{
+ uint8_t t[Nb * Nk];
+
+ mult_row_column(t, s);
+ mult_row_column(&t[Nb], s + Nb);
+ mult_row_column(&t[2 * Nb], s + (2 * Nb));
+ mult_row_column(&t[3 * Nb], s + (3 * Nb));
+ (void) _copy(s, sizeof(t), t, sizeof(t));
+}
+
+/*
+ * This shift_rows also implements the matrix flip required for mix_columns, but
+ * performs it here to reduce the number of memory operations.
+ */
+static inline void shift_rows(uint8_t *s)
+{
+ uint8_t t[Nb * Nk];
+
+ t[0] = s[0]; t[1] = s[5]; t[2] = s[10]; t[3] = s[15];
+ t[4] = s[4]; t[5] = s[9]; t[6] = s[14]; t[7] = s[3];
+ t[8] = s[8]; t[9] = s[13]; t[10] = s[2]; t[11] = s[7];
+ t[12] = s[12]; t[13] = s[1]; t[14] = s[6]; t[15] = s[11];
+ (void) _copy(s, sizeof(t), t, sizeof(t));
+}
+
+int tc_aes_encrypt(uint8_t *out, const uint8_t *in, const TCAesKeySched_t s)
+{
+ uint8_t state[Nk * Nb];
+ unsigned int i;
+
+ if (out == (uint8_t *) 0) {
+ return TC_CRYPTO_FAIL;
+ } else if (in == (const uint8_t *) 0) {
+ return TC_CRYPTO_FAIL;
+ } else if (s == (TCAesKeySched_t) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ (void)_copy(state, sizeof(state), in, sizeof(state));
+ add_round_key(state, s->words);
+
+ for (i = 0; i < (Nr - 1); ++i) {
+ sub_bytes(state);
+ shift_rows(state);
+ mix_columns(state);
+ add_round_key(state, s->words + Nb * (i + 1));
+ }
+
+ sub_bytes(state);
+ shift_rows(state);
+ add_round_key(state, s->words + Nb * (i + 1));
+
+ (void)_copy(out, sizeof(state), state, sizeof(state));
+
+ /* zeroing out the state buffer */
+ _set(state, TC_ZERO_BYTE, sizeof(state));
+
+ return TC_CRYPTO_SUCCESS;
+}
+
+int tc_cmac_setup(TCCmacState_t s, const uint8_t *key, TCAesKeySched_t sched)
+{
+
+ /* input sanity check: */
+ if (s == (TCCmacState_t) 0 ||
+ key == (const uint8_t *) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ /* put s into a known state */
+ _set(s, 0, sizeof(*s));
+ s->sched = sched;
+
+ /* configure the encryption key used by the underlying block cipher */
+ tc_aes128_set_encrypt_key(s->sched, key);
+
+ /* compute s->K1 and s->K2 from s->iv using s->keyid */
+ _set(s->iv, 0, TC_AES_BLOCK_SIZE);
+ tc_aes_encrypt(s->iv, s->iv, s->sched);
+ gf_double (s->K1, s->iv);
+ gf_double (s->K2, s->K1);
+
+ /* reset s->iv to 0 in case someone wants to compute now */
+ tc_cmac_init(s);
+
+ return TC_CRYPTO_SUCCESS;
+}
+
+/*
+ * assumes: out != NULL and points to a GF(2^n) value to receive the
+ * doubled value;
+ * in != NULL and points to a 16 byte GF(2^n) value
+ * to double;
+ * the in and out buffers do not overlap.
+ * effects: doubles the GF(2^n) value pointed to by "in" and places
+ * the result in the GF(2^n) value pointed to by "out."
+ */
+void gf_double(uint8_t *out, uint8_t *in)
+{
+
+ /* start with low order byte */
+ uint8_t *x = in + (TC_AES_BLOCK_SIZE - 1);
+
+ /* if msb == 1, we need to add the gf_wrap value, otherwise add 0 */
+ uint8_t carry = (in[0] >> 7) ? gf_wrap : 0;
+
+ out += (TC_AES_BLOCK_SIZE - 1);
+ for (;;) {
+ *out-- = (*x << 1) ^ carry;
+ if (x == in) {
+ break;
+ }
+ carry = *x-- >> 7;
+ }
+}
+
+int tc_cmac_init(TCCmacState_t s)
+{
+ /* input sanity check: */
+ if (s == (TCCmacState_t) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ /* CMAC starts with an all zero initialization vector */
+ _set(s->iv, 0, TC_AES_BLOCK_SIZE);
+
+ /* and the leftover buffer is empty */
+ _set(s->leftover, 0, TC_AES_BLOCK_SIZE);
+ s->leftover_offset = 0;
+
+ /* Set countdown to max number of calls allowed before re-keying: */
+ s->countdown = MAX_CALLS;
+
+ return TC_CRYPTO_SUCCESS;
+}
+
+int tc_cmac_update(TCCmacState_t s, const uint8_t *data, size_t data_length)
+{
+ unsigned int i;
+
+ /* input sanity check: */
+ if (s == (TCCmacState_t) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+ if (data_length == 0) {
+ return TC_CRYPTO_SUCCESS;
+ }
+ if (data == (const uint8_t *) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ if (s->countdown == 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ s->countdown--;
+
+ if (s->leftover_offset > 0) {
+ /* last data added to s didn't end on a TC_AES_BLOCK_SIZE byte boundary */
+ size_t remaining_space = TC_AES_BLOCK_SIZE - s->leftover_offset;
+
+ if (data_length < remaining_space) {
+ /* still not enough data to encrypt this time either */
+ _copy(&s->leftover[s->leftover_offset], data_length, data, data_length);
+ s->leftover_offset += data_length;
+ return TC_CRYPTO_SUCCESS;
+ }
+ /* leftover block is now full; encrypt it first */
+ _copy(&s->leftover[s->leftover_offset],
+ remaining_space,
+ data,
+ remaining_space);
+ data_length -= remaining_space;
+ data += remaining_space;
+ s->leftover_offset = 0;
+
+ for (i = 0; i < TC_AES_BLOCK_SIZE; ++i) {
+ s->iv[i] ^= s->leftover[i];
+ }
+ tc_aes_encrypt(s->iv, s->iv, s->sched);
+ }
+
+ /* CBC encrypt each (except the last) of the data blocks */
+ while (data_length > TC_AES_BLOCK_SIZE) {
+ for (i = 0; i < TC_AES_BLOCK_SIZE; ++i) {
+ s->iv[i] ^= data[i];
+ }
+ tc_aes_encrypt(s->iv, s->iv, s->sched);
+ data += TC_AES_BLOCK_SIZE;
+ data_length -= TC_AES_BLOCK_SIZE;
+ }
+
+ if (data_length > 0) {
+ /* save leftover data for next time */
+ _copy(s->leftover, data_length, data, data_length);
+ s->leftover_offset = data_length;
+ }
+
+ return TC_CRYPTO_SUCCESS;
+}
+
+int tc_cmac_final(uint8_t *tag, TCCmacState_t s)
+{
+ uint8_t *k;
+ unsigned int i;
+
+ /* input sanity check: */
+ if (tag == (uint8_t *) 0 ||
+ s == (TCCmacState_t) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ if (s->leftover_offset == TC_AES_BLOCK_SIZE) {
+ /* the last message block is a full-sized block */
+ k = (uint8_t *) s->K1;
+ } else {
+ /* the final message block is not a full-sized block */
+ size_t remaining = TC_AES_BLOCK_SIZE - s->leftover_offset;
+
+ _set(&s->leftover[s->leftover_offset], 0, remaining);
+ s->leftover[s->leftover_offset] = TC_CMAC_PADDING;
+ k = (uint8_t *) s->K2;
+ }
+ for (i = 0; i < TC_AES_BLOCK_SIZE; ++i) {
+ s->iv[i] ^= s->leftover[i] ^ k[i];
+ }
+
+ tc_aes_encrypt(tag, s->iv, s->sched);
+
+ /* erasing state: */
+ tc_cmac_erase(s);
+
+ return TC_CRYPTO_SUCCESS;
+}
+
+int tc_cmac_erase(TCCmacState_t s)
+{
+ if (s == (TCCmacState_t) 0) {
+ return TC_CRYPTO_FAIL;
+ }
+
+ /* destroy the current state */
+ _set(s, 0, sizeof(*s));
+
+ return TC_CRYPTO_SUCCESS;
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/mesh_core/mesh_atomic.c b/components/bt/ble_mesh/mesh_core/mesh_atomic.c
new file mode 100644
index 0000000000..a268a792b1
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_atomic.c
@@ -0,0 +1,180 @@
+/**
+ * @brief Atomically set a bit.
+ *
+ * Atomically set bit number @a bit of @a target.
+ * The target may be a single atomic variable or an array of them.
+ *
+ * @param target Address of atomic variable or array.
+ * @param bit Bit number (starting from 0).
+ *
+ * @return N/A
+ */
+
+/*
+ * Copyright (c) 2016 Intel Corporation
+ * Copyright (c) 2011-2014 Wind River Systems, Inc.
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "mesh_atomic.h"
+#include "mesh_kernel.h"
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+
+/**
+*
+* @brief Atomic get primitive
+*
+* @param target memory location to read from
+*
+* This routine provides the atomic get primitive to atomically read
+* a value from . It simply does an ordinary load. Note that
+* is expected to be aligned to a 4-byte boundary.
+*
+* @return The value read from
+*/
+atomic_val_t atomic_get(const atomic_t *target)
+{
+ return *target;
+}
+
+/**
+ *
+ * @brief Atomic get-and-set primitive
+ *
+ * This routine provides the atomic set operator. The is atomically
+ * written at and the previous value at is returned.
+ *
+ * @param target the memory location to write to
+ * @param value the value to write
+ *
+ * @return The previous value from
+ */
+atomic_val_t atomic_set(atomic_t *target, atomic_val_t value)
+{
+ unsigned int key;
+ atomic_val_t ret;
+
+ key = irq_lock();
+
+ ret = *target;
+ *target = value;
+
+ irq_unlock(key);
+
+ return ret;
+}
+
+/**
+ *
+ * @brief Atomic bitwise inclusive OR primitive
+ *
+ * This routine provides the atomic bitwise inclusive OR operator. The
+ * is atomically bitwise OR'ed with the value at , placing the result
+ * at , and the previous value at is returned.
+ *
+ * @param target the memory location to be modified
+ * @param value the value to OR
+ *
+ * @return The previous value from
+ */
+atomic_val_t atomic_or(atomic_t *target, atomic_val_t value)
+{
+ unsigned int key;
+ atomic_val_t ret;
+
+ key = irq_lock();
+
+ ret = *target;
+ *target |= value;
+
+ irq_unlock(key);
+
+ return ret;
+}
+
+/**
+ *
+ * @brief Atomic bitwise AND primitive
+ *
+ * This routine provides the atomic bitwise AND operator. The is
+ * atomically bitwise AND'ed with the value at , placing the result
+ * at , and the previous value at is returned.
+ *
+ * @param target the memory location to be modified
+ * @param value the value to AND
+ *
+ * @return The previous value from
+ */
+atomic_val_t atomic_and(atomic_t *target, atomic_val_t value)
+{
+ unsigned int key;
+ atomic_val_t ret;
+
+ key = irq_lock();
+
+ ret = *target;
+ *target &= value;
+
+ irq_unlock(key);
+
+ return ret;
+}
+
+/**
+ *
+ * @brief Atomic decrement primitive
+ *
+ * @param target memory location to decrement
+ *
+ * This routine provides the atomic decrement operator. The value at
+ * is atomically decremented by 1, and the old value from is returned.
+ *
+ * @return The value from prior to the decrement
+ */
+atomic_val_t atomic_dec(atomic_t *target)
+{
+ unsigned int key;
+ atomic_val_t ret;
+
+ key = irq_lock();
+
+ ret = *target;
+ (*target)--;
+
+ irq_unlock(key);
+
+ return ret;
+}
+
+/**
+ *
+ * @brief Atomic increment primitive
+ *
+ * @param target memory location to increment
+ *
+ * This routine provides the atomic increment operator. The value at
+ * is atomically incremented by 1, and the old value from is returned.
+ *
+ * @return The value from before the increment
+ */
+atomic_val_t atomic_inc(atomic_t *target)
+{
+ unsigned int key;
+ atomic_val_t ret;
+
+ key = irq_lock();
+
+ ret = *target;
+ (*target)++;
+
+ irq_unlock(key);
+
+ return ret;
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/mesh_core/mesh_bearer_adapt.c b/components/bt/ble_mesh/mesh_core/mesh_bearer_adapt.c
new file mode 100644
index 0000000000..f4dae675b8
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_bearer_adapt.c
@@ -0,0 +1,2157 @@
+/*
+ * Copyright (c) 2017 Nordic Semiconductor ASA
+ * Copyright (c) 2015-2016 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+
+#include "mesh_bearer_adapt.h"
+#include "mesh_trace.h"
+
+#include "bta/bta_api.h"
+#include "bta/bta_gatt_api.h"
+#include "bta/bta_gatt_common.h"
+#include "stack/btm_ble_api.h"
+#include "p_256_ecc_pp.h"
+
+#include "stack/hcimsgs.h"
+#include "osi/future.h"
+#include "include/mesh_buf.h"
+#include "osi/allocator.h"
+#include "include/mesh_atomic.h"
+#include "include/mesh_hci.h"
+
+#include "mbedtls/aes.h"
+#include "mesh_aes_encrypt.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+
+#include "bta/bta_api.h"
+#include
+#include "include/mesh_buf.h"
+#include "osi/allocator.h"
+#include "include/mesh_atomic.h"
+#include "bta_gattc_int.h"
+#include "provisioner_prov.h"
+#include "common.h"
+
+#define BLE_MESH_BTM_CHECK_STATUS(func) do { \
+ tBTM_STATUS __status = (func); \
+ if ((__status != BTM_SUCCESS) && (__status != BTM_CMD_STARTED)) { \
+ BT_ERR("%s, invalid status %d", __func__, __status); \
+ return -1; \
+ } \
+ } while(0);
+
+#define BT_MESH_GATT_GET_CONN_ID(conn_id) (((u16_t)(conn_id)) >> 8)
+#define BT_MESH_GATT_CREATE_CONN_ID(gatt_if, conn_id) ((u16_t)((((u8_t)(conn_id)) << 8) | ((u8_t)(gatt_if))))
+#define CONFIG_BT_MAX_CONN CONFIG_BT_ACL_CONNECTIONS
+
+/* We don't need to manage the BT_DEV_ADVERTISING flags in the version of bluedriod,
+ * it will manage it in the BTM layer.
+ */
+#define BT_DEV 0
+
+/* Global Variables */
+extern struct bt_dev bt_dev;
+
+/* P-256 Variables */
+static u8_t bt_mesh_public_key[64];
+static BT_OCTET32 bt_mesh_private_key = {
+ 0x3f, 0x49, 0xf6, 0xd4, 0xa3, 0xc5, 0x5f, 0x38,
+ 0x74, 0xc9, 0xb3, 0xe3, 0xd2, 0x10, 0x3f, 0x50,
+ 0x4a, 0xff, 0x60, 0x7b, 0xeb, 0x40, 0xb7, 0x99,
+ 0x58, 0x99, 0xb8, 0xa6, 0xcd, 0x3c, 0x1a, 0xbd
+};
+
+/* Scan related functions */
+static bt_le_scan_cb_t *bt_mesh_scan_dev_found_cb;
+static void bt_mesh_scan_result_callback(tBTA_DM_SEARCH_EVT event, tBTA_DM_SEARCH *p_data);
+
+#if defined(CONFIG_BT_MESH_NODE) && CONFIG_BT_MESH_NODE
+/* the gatt database list to save the attribute table */
+static sys_slist_t bt_mesh_gatts_db;
+
+/* Static Variables */
+static struct bt_conn bt_mesh_gatts_conn[CONFIG_BT_MAX_CONN];
+static struct bt_conn_cb *bt_mesh_gatts_conn_cb;
+static tBTA_GATTS_IF bt_mesh_gatts_if;
+static u16_t svc_handle, char_handle;
+static future_t *future_mesh;
+
+/* Static Functions */
+static struct bt_gatt_attr *bt_mesh_gatts_find_attr_by_handle(u16_t handle);
+#endif /* defined(CONFIG_BT_MESH_NODE) && CONFIG_BT_MESH_NODE */
+
+#if defined(CONFIG_BT_MESH_PROVISIONER) && CONFIG_BT_MESH_PROVISIONER
+#define BT_MESH_GATTC_APP_UUID_BYTE 0x97
+static struct gattc_prov_info {
+ /* Service to be found depends on the type of adv pkt received */
+ struct bt_conn conn;
+ BD_ADDR addr;
+ u8_t addr_type;
+ u16_t service_uuid;
+ u16_t mtu;
+ bool wr_desc_done; /* Indicate if write char descriptor event is received */
+ u16_t start_handle; /* Service attribute start handle */
+ u16_t end_handle; /* Service attribute end handle */
+ u16_t data_in_handle; /* Data In Characteristic attribute handle */
+ u16_t data_out_handle; /* Data Out Characteristic attribute handle */
+ u16_t ccc_handle; /* Data Out Characteristic CCC attribute handle */
+} bt_mesh_gattc_info[CONFIG_BT_MAX_CONN];
+static struct bt_prov_conn_cb *bt_mesh_gattc_conn_cb;
+static tBTA_GATTC_IF bt_mesh_gattc_if;
+#endif /* defined(CONFIG_BT_MESH_PROVISIONER) && CONFIG_BT_MESH_PROVISIONER */
+
+static void ble_mesh_scan_results_change_2_bta(tBTM_INQ_RESULTS *p_inq, u8_t *p_eir, tBTA_DM_SEARCH_CBACK *p_scan_cback)
+{
+ tBTA_DM_SEARCH result;
+ tBTM_INQ_INFO *p_inq_info;
+
+ bdcpy(result.inq_res.bd_addr, p_inq->remote_bd_addr);
+ result.inq_res.rssi = p_inq->rssi;
+ result.inq_res.ble_addr_type = p_inq->ble_addr_type;
+ result.inq_res.inq_result_type = p_inq->inq_result_type;
+ result.inq_res.device_type = p_inq->device_type;
+ result.inq_res.flag = p_inq->flag;
+ result.inq_res.adv_data_len = p_inq->adv_data_len;
+ result.inq_res.scan_rsp_len = p_inq->scan_rsp_len;
+ memcpy(result.inq_res.dev_class, p_inq->dev_class, sizeof(DEV_CLASS));
+ result.inq_res.ble_evt_type = p_inq->ble_evt_type;
+
+ /* application will parse EIR to find out remote device name */
+ result.inq_res.p_eir = p_eir;
+
+ if ((p_inq_info = BTM_InqDbRead(p_inq->remote_bd_addr)) != NULL) {
+ /* initialize remt_name_not_required to FALSE so that we get the name by default */
+ result.inq_res.remt_name_not_required = FALSE;
+ }
+
+ if (p_scan_cback) {
+ p_scan_cback(BTA_DM_INQ_RES_EVT, &result);
+ }
+
+ if (p_inq_info) {
+ /* application indicates if it knows the remote name, inside the callback
+ copy that to the inquiry data base*/
+ if (result.inq_res.remt_name_not_required) {
+ p_inq_info->appl_knows_rem_name = TRUE;
+ }
+ }
+}
+
+static void ble_mesh_scan_results_cb(tBTM_INQ_RESULTS *p_inq, u8_t *p_eir)
+{
+ ble_mesh_scan_results_change_2_bta(p_inq, p_eir, bt_mesh_scan_result_callback);
+}
+
+static bool valid_adv_param(const struct bt_le_adv_param *param)
+{
+ if (!(param->options & BT_LE_ADV_OPT_CONNECTABLE)) {
+ /*
+ * BT Core 4.2 [Vol 2, Part E, 7.8.5]
+ * The Advertising_Interval_Min and Advertising_Interval_Max
+ * shall not be set to less than 0x00A0 (100 ms) if the
+ * Advertising_Type is set to ADV_SCAN_IND or ADV_NONCONN_IND.
+ */
+#if BT_DEV
+ /**TODO: Need to optimize controller to make it supports 20ms interval */
+ if (bt_dev.hci_version < BT_HCI_VERSION_5_0 &&
+ param->interval_min < 0x00a0) {
+ return false;
+ }
+#endif
+ }
+
+ if (param->interval_min > param->interval_max ||
+ param->interval_min < 0x0020 || param->interval_max > 0x4000) {
+ return false;
+ }
+
+ return true;
+}
+
+static int set_ad(u16_t hci_op, const struct bt_data *ad, size_t ad_len)
+{
+ struct bt_hci_cp_le_set_adv_data set_data_param;
+ struct bt_hci_cp_le_set_adv_data *set_data = &set_data_param;
+ int i;
+
+#if 0
+ struct net_buf *buf;
+ buf = bt_hci_cmd_create(hci_op, sizeof(*set_data));
+ if (!buf) {
+ return -ENOBUFS;
+ }
+#endif
+
+ memset(set_data, 0, sizeof(*set_data));
+
+ for (i = 0; i < ad_len; i++) {
+ /* Check if ad fit in the remaining buffer */
+ if (set_data->len + ad[i].data_len + 2 > 31) {
+#if 0
+ net_buf_unref(buf);
+#endif
+ return -EINVAL;
+ }
+
+ set_data->data[set_data->len++] = ad[i].data_len + 1;
+ set_data->data[set_data->len++] = ad[i].type;
+
+ memcpy(&set_data->data[set_data->len], ad[i].data, ad[i].data_len);
+ set_data->len += ad[i].data_len;
+ }
+
+ /* Set adv data and scan rsp data, TODO: how to process null adv data?
+ * Do we need to set adv data every time even though the adv data is not changed?
+ */
+
+#if 0
+ return bt_hci_cmd_send_sync(hci_op, buf, NULL);
+#else
+ if (hci_op == BT_HCI_OP_LE_SET_ADV_DATA) {
+ BLE_MESH_BTM_CHECK_STATUS(BTM_BleWriteAdvDataRaw(set_data->data, set_data->len));
+ } else if (hci_op == BT_HCI_OP_LE_SET_SCAN_RSP_DATA) {
+ BLE_MESH_BTM_CHECK_STATUS(BTM_BleWriteScanRspRaw(set_data->data, set_data->len));
+ }
+ return 0;
+#endif /* #if 0 */
+}
+
+static void start_adv_completed_cb(u8_t status)
+{
+ /**TODO: It is too late to wait for completed event*/
+#if 0
+ if (!status) {
+ atomic_set_bit(bt_dev.flags, BT_DEV_ADVERTISING);
+ }
+#endif
+}
+
+static bool valid_le_scan_param(const struct bt_le_scan_param *param)
+{
+ if (param->type != BT_HCI_LE_SCAN_PASSIVE &&
+ param->type != BT_HCI_LE_SCAN_ACTIVE) {
+ return false;
+ }
+
+ if (param->filter_dup != BT_HCI_LE_SCAN_FILTER_DUP_DISABLE &&
+ param->filter_dup != BT_HCI_LE_SCAN_FILTER_DUP_ENABLE) {
+ return false;
+ }
+
+ if (param->interval < 0x0004 || param->interval > 0x4000) {
+ return false;
+ }
+
+ if (param->window < 0x0004 || param->window > 0x4000) {
+ return false;
+ }
+
+ if (param->window > param->interval) {
+ return false;
+ }
+
+ return true;
+}
+
+static int start_le_scan(u8_t scan_type, u16_t interval, u16_t window)
+{
+ int err = 0;
+
+#if 0
+ struct bt_hci_cp_le_set_scan_param set_param;
+ struct net_buf *buf;
+ memset(&set_param, 0, sizeof(set_param));
+
+ set_param.scan_type = scan_type;
+ /* for the rest parameters apply default values according to
+ * spec 4.2, vol2, part E, 7.8.10
+ */
+ set_param.interval = sys_cpu_to_le16(interval);
+ set_param.window = sys_cpu_to_le16(window);
+ set_param.filter_policy = 0x00;
+#endif
+
+ /* TODO: Currently support only RPA and set only once before scan */
+#if 0
+ if (IS_ENABLED(CONFIG_BT_PRIVACY)) {
+ err = le_set_private_addr();
+ if (err) {
+ return err;
+ }
+
+ if (BT_FEAT_LE_PRIVACY(bt_dev.le.features)) {
+ set_param.addr_type = BT_HCI_OWN_ADDR_RPA_OR_RANDOM;
+ } else {
+ set_param.addr_type = BT_ADDR_LE_RANDOM;
+ }
+ } else {
+ set_param.addr_type = bt_dev.id_addr.type;
+
+ /* Use NRPA unless identity has been explicitly requested
+ * (through Kconfig), or if there is no advertising ongoing.
+ */
+ if (!IS_ENABLED(CONFIG_BT_SCAN_WITH_IDENTITY) &&
+ scan_type == BT_HCI_LE_SCAN_ACTIVE &&
+ !atomic_test_bit(bt_dev.flags, BT_DEV_ADVERTISING)) {
+ err = le_set_private_addr();
+ if (err) {
+ return err;
+ }
+
+ set_param.addr_type = BT_ADDR_LE_RANDOM;
+ }
+ }
+#endif
+
+#if 0
+ buf = bt_hci_cmd_create(BT_HCI_OP_LE_SET_SCAN_PARAM, sizeof(set_param));
+ if (!buf) {
+ return -ENOBUFS;
+ }
+
+ net_buf_add_mem(buf, &set_param, sizeof(set_param));
+
+ bt_hci_cmd_send(BT_HCI_OP_LE_SET_SCAN_PARAM, buf);
+
+ err = set_le_scan_enable(BT_HCI_LE_SCAN_ENABLE);
+ if (err) {
+ return err;
+ }
+#else
+ tGATT_IF client_if = 0xFF; //default GATT interface id
+ UINT8 scan_fil_policy = 0x00; //No white-list for mesh
+ UINT8 addr_type_own = 0x0; //TODO: currently support only RPA
+
+ /*TODO: Need to process scan_param_setup_cback
+ * Need to add menuconfig for duplicate scan*/
+#if defined(CONFIG_BLE_MESH_SCAN_DUPLICATE_EN) && CONFIG_BLE_MESH_SCAN_DUPLICATE_EN
+ BTM_BleSetScanFilterParams(client_if, interval, window, scan_type, addr_type_own,
+ BT_HCI_LE_SCAN_FILTER_DUP_ENABLE, scan_fil_policy, NULL);
+#else
+ BTM_BleSetScanFilterParams(client_if, interval, window, scan_type, addr_type_own,
+ BT_HCI_LE_SCAN_FILTER_DUP_DISABLE, scan_fil_policy, NULL);
+#endif
+
+ /*TODO: Need to process p_start_stop_scan_cb to check if start successfully */
+ /* BLE Mesh scan permanently, so no duration of scan here */
+ BLE_MESH_BTM_CHECK_STATUS(BTM_BleScan(true, 0, ble_mesh_scan_results_cb, NULL, NULL));
+#endif /* #if 0 */
+
+#if BT_DEV
+ if (scan_type == BT_HCI_LE_SCAN_ACTIVE) {
+ atomic_set_bit(bt_dev.flags, BT_DEV_ACTIVE_SCAN);
+ } else {
+ atomic_clear_bit(bt_dev.flags, BT_DEV_ACTIVE_SCAN);
+ }
+#endif
+
+ return err;
+}
+
+static void bt_mesh_scan_result_callback(tBTA_DM_SEARCH_EVT event, tBTA_DM_SEARCH *p_data)
+{
+ bt_addr_le_t addr;
+ UINT8 rssi;
+ UINT8 adv_type;
+
+ BT_DBG("%s, event = %d", __func__, event);
+
+ if (event == BTA_DM_INQ_RES_EVT) {
+ /* TODO: How to process scan_rsp here? */
+ addr.type = p_data->inq_res.ble_addr_type;
+ memcpy(addr.a.val, p_data->inq_res.bd_addr, BD_ADDR_LEN);
+ rssi = p_data->inq_res.rssi;
+ adv_type = p_data->inq_res.ble_evt_type;
+
+ //scan rsp len: p_data->inq_res.scan_rsp_len
+ struct net_buf_simple *buf = bt_mesh_alloc_buf(p_data->inq_res.adv_data_len);
+ if (!buf) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return;
+ }
+ net_buf_simple_init(buf, 0);
+ net_buf_simple_add_mem(buf, p_data->inq_res.p_eir, p_data->inq_res.adv_data_len);
+
+ if (bt_mesh_scan_dev_found_cb != NULL) {
+ bt_mesh_scan_dev_found_cb(&addr, rssi, adv_type, buf);
+ }
+ osi_free(buf);
+ } else if (event == BTA_DM_INQ_CMPL_EVT) {
+ BT_INFO("%s, Scan completed, number of scan response %d", __func__, p_data->inq_cmpl.num_resps);
+ } else {
+ BT_WARN("%s, Unexpected event 0x%x", __func__, event);
+ }
+}
+
+int bt_le_scan_update(bool fast_scan)
+{
+#if BT_DEV
+ if (atomic_test_bit(bt_dev.flags, BT_DEV_EXPLICIT_SCAN)) {
+ return 0;
+ }
+#endif /* #if BT_DEV */
+
+#if BT_DEV
+ if (atomic_test_bit(bt_dev.flags, BT_DEV_SCANNING)) {
+#else /* #if BT_DEV */
+ if (1) {
+#endif /* #if BT_DEV */
+#if 0
+ int err = set_le_scan_enable(BT_HCI_LE_SCAN_DISABLE);
+ if (err) {
+ return err;
+ }
+#else /* #if 0 */
+ /* TODO: need to process stop_scan_cb */
+#if BT_DEV
+ atomic_clear_bit(bt_dev.flags, BT_DEV_SCANNING);
+#endif
+ BLE_MESH_BTM_CHECK_STATUS(BTM_BleScan(false, 0, NULL, NULL, NULL));
+#endif /* #if 0 */
+ }
+
+ /* We don't need to stay pasive scan for central device */
+#if 0
+ if (IS_ENABLED(CONFIG_BT_CENTRAL)) {
+ u16_t interval, window;
+ struct bt_conn *conn;
+
+ conn = bt_conn_lookup_state_le(NULL, BT_CONN_CONNECT_SCAN);
+ if (!conn) {
+ return 0;
+ }
+
+ atomic_set_bit(bt_dev.flags, BT_DEV_SCAN_FILTER_DUP);
+
+ bt_mesh_conn_unref(conn);
+
+ if (fast_scan) {
+ interval = BT_GAP_SCAN_FAST_INTERVAL;
+ window = BT_GAP_SCAN_FAST_WINDOW;
+ } else {
+ interval = BT_GAP_SCAN_SLOW_INTERVAL_1;
+ window = BT_GAP_SCAN_SLOW_WINDOW_1;
+ }
+
+ return start_le_scan(BT_HCI_LE_SCAN_PASSIVE, interval, window);
+ }
+#endif /* #if 0 */
+
+ return 0;
+}
+
+/* APIs functions */
+int bt_le_adv_start(const struct bt_le_adv_param *param,
+ const struct bt_data *ad, size_t ad_len,
+ const struct bt_data *sd, size_t sd_len)
+{
+ /** TODO:
+ *- Need to support 20ms interval in 4.2
+ *- Need to check adv start HCI event
+ *- Random address check, currently we set privacy only once in application
+ *- Process ADV_OPT_ONE_TIME
+ **/
+ UINT16 adv_int_min;
+ UINT16 adv_int_max;
+ tBLE_ADDR_TYPE addr_type_own;
+ UINT8 adv_type;
+ tBTM_BLE_ADV_CHNL_MAP chnl_map;
+ tBTM_BLE_AFP adv_fil_pol;
+ tBLE_BD_ADDR p_dir_bda;
+ tBTA_START_ADV_CMPL_CBACK *p_start_adv_cb;
+
+ struct bt_hci_cp_le_set_adv_param set_param;
+ int err = 0;
+
+ if (!valid_adv_param(param)) {
+ return -EINVAL;
+ }
+
+#if BT_DEV
+#if 0
+ /**TODO: We cancel this check temporarily*/
+ if (atomic_test_bit(bt_dev.flags, BT_DEV_ADVERTISING)) {
+ return -EALREADY;
+ }
+#endif /* #if 0 */
+#endif /* #if BT_DEV */
+
+ err = set_ad(BT_HCI_OP_LE_SET_ADV_DATA, ad, ad_len);
+ if (err) {
+ return err;
+ }
+
+ /*
+ * We need to set SCAN_RSP when enabling advertising type that allows
+ * for Scan Requests.
+ *
+ * If sd was not provided but we enable connectable undirected
+ * advertising sd needs to be cleared from values set by previous calls.
+ * Clearing sd is done by calling set_ad() with NULL data and zero len.
+ * So following condition check is unusual but correct.
+ */
+ if (sd || (param->options & BT_LE_ADV_OPT_CONNECTABLE)) {
+ err = set_ad(BT_HCI_OP_LE_SET_SCAN_RSP_DATA, sd, sd_len);
+ if (err) {
+ return err;
+ }
+ }
+
+ memset(&set_param, 0, sizeof(set_param));
+
+ set_param.min_interval = sys_cpu_to_le16(param->interval_min);
+ set_param.max_interval = sys_cpu_to_le16(param->interval_max);
+ set_param.channel_map = 0x07;
+
+ /** TODO: Currently we support only RPA address */
+#if 0
+ if (param->options & BT_LE_ADV_OPT_CONNECTABLE) {
+ if (IS_ENABLED(CONFIG_BT_PRIVACY)) {
+ err = le_set_private_addr();
+ if (err) {
+ return err;
+ }
+
+ if (BT_FEAT_LE_PRIVACY(bt_dev.le.features)) {
+ set_param.own_addr_type =
+ BT_HCI_OWN_ADDR_RPA_OR_RANDOM;
+ } else {
+ set_param.own_addr_type = BT_ADDR_LE_RANDOM;
+ }
+ } else {
+ /*
+ * If Static Random address is used as Identity
+ * address we need to restore it before advertising
+ * is enabled. Otherwise NRPA used for active scan
+ * could be used for advertising.
+ */
+ if (atomic_test_bit(bt_dev.flags,
+ BT_DEV_ID_STATIC_RANDOM)) {
+ set_random_address(&bt_dev.id_addr.a);
+ }
+
+ set_param.own_addr_type = bt_dev.id_addr.type;
+ }
+
+ set_param.type = BT_LE_ADV_IND;
+ } else {
+ if (param->own_addr) {
+ /* Only NRPA is allowed */
+ if (!BT_ADDR_IS_NRPA(param->own_addr)) {
+ return -EINVAL;
+ }
+
+ err = set_random_address(param->own_addr);
+ } else {
+ err = le_set_private_addr();
+ }
+
+ if (err) {
+ return err;
+ }
+
+ set_param.own_addr_type = BT_ADDR_LE_RANDOM;
+
+ if (sd) {
+ set_param.type = BT_LE_ADV_SCAN_IND;
+ } else {
+ set_param.type = BT_LE_ADV_NONCONN_IND;
+ }
+ }
+#else /* #if 0 */
+ /* TODO: Has been simplified here, currently support only RPA addr */
+ set_param.own_addr_type = 0x0; //BT_ADDR_LE_RANDOM
+ if (param->options & BT_LE_ADV_OPT_CONNECTABLE) {
+ set_param.type = 0x00; //ADV_TYPE_IND;
+ } else if (sd != NULL) {
+ set_param.type = 0x02; //ADV_TYPE_SCAN_IND;
+ } else {
+ set_param.type = 0x03; //ADV_TYPE_NONCONN_IND;
+ }
+#endif /* #if 0 */
+
+#if 0
+ buf = bt_hci_cmd_create(BT_HCI_OP_LE_SET_ADV_PARAM, sizeof(set_param));
+ if (!buf) {
+ return -ENOBUFS;
+ }
+
+ net_buf_add_mem(buf, &set_param, sizeof(set_param));
+
+ err = bt_hci_cmd_send_sync(BT_HCI_OP_LE_SET_ADV_PARAM, buf, NULL);
+ if (err) {
+ return err;
+ }
+
+ err = set_advertise_enable(true);
+ if (err) {
+ return err;
+ }
+#else /* #if 0 */
+ addr_type_own = set_param.own_addr_type;
+ /* Set adv parameters */
+ adv_int_min = param->interval_min;
+ adv_int_max = param->interval_max;
+ adv_type = set_param.type;
+ chnl_map = BTA_BLE_ADV_CHNL_37 | BTA_BLE_ADV_CHNL_38 | BTA_BLE_ADV_CHNL_39;
+ adv_fil_pol = AP_SCAN_CONN_ALL;
+ memset(&p_dir_bda, 0, sizeof(p_dir_bda));
+ p_start_adv_cb = start_adv_completed_cb;
+
+ /* TODO: We need to add a check function, to make sure adv start successfully and then set bit
+ * So currently we call BTM_BleSetAdvParamsStartAdv instead of BTA_DmSetBleAdvParamsAll
+ */
+#if 1
+ /* Check if we can start adv using BTM_BleSetAdvParamsStartAdvCheck */
+ BLE_MESH_BTM_CHECK_STATUS(BTM_BleSetAdvParamsAll(adv_int_min, adv_int_max, adv_type,
+ addr_type_own, &p_dir_bda,
+ chnl_map, adv_fil_pol, p_start_adv_cb));
+ BLE_MESH_BTM_CHECK_STATUS(BTM_BleStartAdv());
+#if BT_DEV
+ atomic_set_bit(bt_dev.flags, BT_DEV_ADVERTISING);
+#endif /* #if BT_DEV */
+#else /* #if 1 */
+ status = BTM_BleSetAdvParamsStartAdv (adv_int_min, adv_int_max, adv_type, addr_type_own, &p_dir_bda, chnl_map, adv_fil_pol, p_start_adv_cb);
+ if (!status) {
+ atomic_set_bit(bt_dev.flags, BT_DEV_ADVERTISING);
+ }
+#endif /* #if 1 */
+#endif /* #if 0 */
+
+ /** TODO: To implement BT_LE_ADV_OPT_ONE_TIME */
+#if BT_DEV
+ if (!(param->options & BT_LE_ADV_OPT_ONE_TIME)) {
+ atomic_set_bit(bt_dev.flags, BT_DEV_KEEP_ADVERTISING);
+ }
+#endif /* #if BT_DEV */
+
+ return err;
+}
+
+int bt_le_adv_stop(void)
+{
+ int err = 0;
+ UINT8 status;
+
+ /* Make sure advertising is not re-enabled later even if it's not
+ * currently enabled (i.e. BT_DEV_ADVERTISING is not set).
+ */
+#if BT_DEV
+ atomic_clear_bit(bt_dev.flags, BT_DEV_KEEP_ADVERTISING);
+
+#if 0
+ /**TODO: We cancel this check temporarily*/
+ if (!atomic_test_bit(bt_dev.flags, BT_DEV_ADVERTISING)) {
+ return 0;
+ }
+#endif /* #if 0 */
+#endif /* #if BT_DEV */
+
+#if 0
+ err = set_advertise_enable(false);
+ if (err) {
+ return err;
+ }
+#else
+ /*TODO: We need to add a check function, to make sure adv start successfully and then set bit
+ * So currently we use BTM_BleBroadcast instead of BTA_DmBleBroadcast*/
+ status = BTM_BleBroadcast(false, NULL);
+ if ((status != BTM_SUCCESS) && (status != BTM_CMD_STARTED)) {
+ BT_ERR("%s, invalid status %d", __func__, status);
+#if BT_DEV
+ atomic_clear_bit(bt_dev.flags, BT_DEV_ADVERTISING);
+#endif
+ return -1;
+ }
+#endif /* #if 0 */
+
+#if 0
+ /* TODO: Currently support RPA only */
+ if (!IS_ENABLED(CONFIG_BT_PRIVACY)) {
+ /* If active scan is ongoing set NRPA */
+ if (atomic_test_bit(bt_dev.flags, BT_DEV_SCANNING) &&
+ atomic_test_bit(bt_dev.flags, BT_DEV_ACTIVE_SCAN)) {
+ le_set_private_addr();
+ }
+ }
+#endif
+
+ return err;
+}
+
+int bt_le_scan_start(const struct bt_le_scan_param *param, bt_le_scan_cb_t cb)
+{
+ /* TODO:
+ * Do we need to use duplicate mode for mesh scan?
+ */
+ int err = 0;
+
+ /* Check that the parameters have valid values */
+ if (!valid_le_scan_param(param)) {
+ return -EINVAL;
+ }
+
+ /* Return if active scan is already enabled */
+#if BT_DEV
+ if (atomic_test_and_set_bit(bt_dev.flags, BT_DEV_EXPLICIT_SCAN)) {
+ return -EALREADY;
+ }
+#endif
+
+ /* TODO: Currently we use bluedroid APIs, which will stop scan automatically
+ * while setting parameters.
+ */
+#if 0
+ if (atomic_test_bit(bt_dev.flags, BT_DEV_SCANNING)) {
+ err = set_le_scan_enable(BT_HCI_LE_SCAN_DISABLE);
+ if (err) {
+ atomic_clear_bit(bt_dev.flags, BT_DEV_EXPLICIT_SCAN);
+ return err;
+ }
+ }
+#endif
+
+#if BT_DEV
+ if (param->filter_dup) {
+ atomic_set_bit(bt_dev.flags, BT_DEV_SCAN_FILTER_DUP);
+ } else {
+ atomic_clear_bit(bt_dev.flags, BT_DEV_SCAN_FILTER_DUP);
+ }
+#endif
+
+ err = start_le_scan(param->type, param->interval, param->window);
+ if (err) {
+#if BT_DEV
+ atomic_clear_bit(bt_dev.flags, BT_DEV_EXPLICIT_SCAN);
+#endif
+ return err;
+ }
+
+#if BT_DEV
+ atomic_set_bit(bt_dev.flags, BT_DEV_SCANNING);
+#endif
+
+ bt_mesh_scan_dev_found_cb = cb;
+ return err;
+}
+
+int bt_le_scan_stop(void)
+{
+ /* Return if active scanning is already disabled */
+#if BT_DEV
+ if (!atomic_test_and_clear_bit(bt_dev.flags, BT_DEV_EXPLICIT_SCAN)) {
+ return -EALREADY;
+ }
+#endif
+
+ bt_mesh_scan_dev_found_cb = NULL;
+
+ return bt_le_scan_update(false);
+}
+
+#if defined(CONFIG_BT_MESH_NODE) && CONFIG_BT_MESH_NODE
+static void bt_mesh_bta_gatts_cb(tBTA_GATTS_EVT event, tBTA_GATTS *p_data)
+{
+ switch (event) {
+ case BTA_GATTS_REG_EVT:
+ if (p_data->reg_oper.status == BTA_GATT_OK) {
+ bt_mesh_gatts_if = p_data->reg_oper.server_if;
+ }
+ break;
+ case BTA_GATTS_READ_EVT: {
+ u8_t buf[100] = {0};
+ u16_t len = 0;
+ tBTA_GATTS_RSP rsp;
+ BT_DBG("%s, read: handle = %d", __func__, p_data->req_data.p_data->read_req.handle);
+ u8_t index = BT_MESH_GATT_GET_CONN_ID(p_data->req_data.conn_id);
+ struct bt_gatt_attr *mesh_attr = bt_mesh_gatts_find_attr_by_handle(p_data->req_data.p_data->read_req.handle);
+ if (mesh_attr != NULL && mesh_attr->read != NULL) {
+ if ((len = mesh_attr->read(&bt_mesh_gatts_conn[index], mesh_attr, buf, 100,
+ p_data->req_data.p_data->read_req.offset)) > 0) {
+ rsp.attr_value.handle = p_data->req_data.p_data->read_req.handle;
+ rsp.attr_value.len = len;
+ memcpy(&rsp.attr_value.value[0], buf, len);
+ BTA_GATTS_SendRsp(p_data->req_data.conn_id, p_data->req_data.trans_id,
+ p_data->req_data.status, &rsp);
+ BT_DBG("%s, send mesh read rsp, handle = %x", __func__, mesh_attr->handle);
+ } else {
+ BT_WARN("%s, read perm Invalid", __func__);
+ }
+ }
+ break;
+ }
+ case BTA_GATTS_WRITE_EVT: {
+ u16_t len = 0;
+ BT_DBG("%s, write: handle = %d, len = %d, data = %s", __func__, p_data->req_data.p_data->write_req.handle,
+ p_data->req_data.p_data->write_req.len,
+ bt_hex(p_data->req_data.p_data->write_req.value, p_data->req_data.p_data->write_req.len));
+ u8_t index = BT_MESH_GATT_GET_CONN_ID(p_data->req_data.conn_id);
+ struct bt_gatt_attr *mesh_attr = bt_mesh_gatts_find_attr_by_handle(p_data->req_data.p_data->write_req.handle);
+ if (mesh_attr != NULL && mesh_attr->write != NULL) {
+ if ((len = mesh_attr->write(&bt_mesh_gatts_conn[index], mesh_attr,
+ p_data->req_data.p_data->write_req.value,
+ p_data->req_data.p_data->write_req.len,
+ p_data->req_data.p_data->write_req.offset, 0)) > 0) {
+ if (p_data->req_data.p_data->write_req.need_rsp) {
+ BTA_GATTS_SendRsp(p_data->req_data.conn_id, p_data->req_data.trans_id,
+ p_data->req_data.status, NULL);
+ BT_DBG("%s, send mesh write rsp, handle = %x", __func__, mesh_attr->handle);
+ }
+ }
+ }
+ break;
+ }
+ case BTA_GATTS_EXEC_WRITE_EVT:
+ break;
+ case BTA_GATTS_MTU_EVT:
+ break;
+ case BTA_GATTS_CONF_EVT:
+ break;
+ case BTA_GATTS_CREATE_EVT: {
+ svc_handle = p_data->create.service_id;
+ BT_DBG("%s, svc_handle = %d, future_mesh = %p", __func__, svc_handle, future_mesh);
+ if (future_mesh != NULL) {
+ future_ready(future_mesh, FUTURE_SUCCESS);
+ }
+ break;
+ }
+ case BTA_GATTS_ADD_INCL_SRVC_EVT: {
+ svc_handle = p_data->add_result.attr_id;
+ if (future_mesh != NULL) {
+ future_ready(future_mesh, FUTURE_SUCCESS);
+ }
+ break;
+ }
+ case BTA_GATTS_ADD_CHAR_EVT:
+ char_handle = p_data->add_result.attr_id;
+ if (future_mesh != NULL) {
+ future_ready(future_mesh, FUTURE_SUCCESS);
+ }
+ break;
+ case BTA_GATTS_ADD_CHAR_DESCR_EVT:
+ char_handle = p_data->add_result.attr_id;
+ if (future_mesh != NULL) {
+ future_ready(future_mesh, FUTURE_SUCCESS);
+ }
+ break;
+ case BTA_GATTS_DELELTE_EVT:
+ break;
+ case BTA_GATTS_START_EVT:
+ break;
+ case BTA_GATTS_STOP_EVT:
+ break;
+ case BTA_GATTS_CONNECT_EVT: {
+ /*Adv disabled*/
+ // atomic_clear_bit(bt_dev.flags, BT_DEV_ADVERTISING);
+ if (bt_mesh_gatts_conn_cb != NULL && bt_mesh_gatts_conn_cb->connected != NULL) {
+ u8_t index = BT_MESH_GATT_GET_CONN_ID(p_data->conn.conn_id);
+ if (index < CONFIG_BT_MAX_CONN) {
+ bt_mesh_gatts_conn[index].handle = BT_MESH_GATT_GET_CONN_ID(p_data->conn.conn_id);
+ (bt_mesh_gatts_conn_cb->connected)(&bt_mesh_gatts_conn[index], 0);
+ }
+ }
+ break;
+ }
+ case BTA_GATTS_DISCONNECT_EVT: {
+ /*Adv disabled*/
+ // atomic_clear_bit(bt_dev.flags, BT_DEV_ADVERTISING);
+ if (bt_mesh_gatts_conn_cb != NULL && bt_mesh_gatts_conn_cb->disconnected != NULL) {
+ u8_t index = BT_MESH_GATT_GET_CONN_ID(p_data->conn.conn_id);
+ if (index < CONFIG_BT_MAX_CONN) {
+ bt_mesh_gatts_conn[index].handle = BT_MESH_GATT_GET_CONN_ID(p_data->conn.conn_id);
+ (bt_mesh_gatts_conn_cb->disconnected)(&bt_mesh_gatts_conn[index], p_data->conn.reason);
+ }
+ }
+ break;
+ }
+ case BTA_GATTS_CLOSE_EVT:
+ break;
+ default:
+ break;
+ }
+}
+
+void bt_mesh_gatts_conn_cb_register(struct bt_conn_cb *cb)
+{
+ bt_mesh_gatts_conn_cb = cb;
+}
+
+static struct bt_gatt_attr *bt_mesh_gatts_find_attr_by_handle(u16_t handle)
+{
+ struct bt_gatt_service *svc;
+ struct bt_gatt_attr *attr = NULL;
+ SYS_SLIST_FOR_EACH_CONTAINER(&bt_mesh_gatts_db, svc, node) {
+ int i;
+
+ for (i = 0; i < svc->attr_count; i++) {
+ attr = &svc->attrs[i];
+ /* Check the attrs handle is equal to the handle or not */
+ if (attr->handle == handle) {
+ return attr;
+ }
+ }
+ }
+
+ return NULL;
+}
+
+static void bt_mesh_gatts_foreach_attr(u16_t start_handle, u16_t end_handle,
+ bt_gatt_attr_func_t func, void *user_data)
+{
+ struct bt_gatt_service *svc;
+
+ SYS_SLIST_FOR_EACH_CONTAINER(&bt_mesh_gatts_db, svc, node) {
+ int i;
+
+ for (i = 0; i < svc->attr_count; i++) {
+ struct bt_gatt_attr *attr = &svc->attrs[i];
+
+ /* Check if attribute handle is within range */
+ if (attr->handle < start_handle ||
+ attr->handle > end_handle) {
+ continue;
+ }
+
+ if (func(attr, user_data) == BT_GATT_ITER_STOP) {
+ return;
+ }
+ }
+ }
+}
+
+static u8_t find_next(const struct bt_gatt_attr *attr, void *user_data)
+{
+ struct bt_gatt_attr **next = user_data;
+
+ *next = (struct bt_gatt_attr *)attr;
+
+ return BT_GATT_ITER_STOP;
+}
+
+static struct bt_gatt_attr *bt_mesh_gatts_attr_next(const struct bt_gatt_attr *attr)
+{
+ struct bt_gatt_attr *next = NULL;
+
+ bt_mesh_gatts_foreach_attr(attr->handle + 1, attr->handle + 1, find_next, &next);
+
+ return next;
+}
+
+ssize_t bt_mesh_gatts_attr_read(struct bt_conn *conn, const struct bt_gatt_attr *attr,
+ void *buf, u16_t buf_len, u16_t offset,
+ const void *value, u16_t value_len)
+{
+ u16_t len;
+
+ if (offset > value_len) {
+ return BT_GATT_ERR(BT_ATT_ERR_INVALID_OFFSET);
+ }
+
+ len = min(buf_len, value_len - offset);
+
+ BT_DBG("handle 0x%04x offset %u length %u", attr->handle, offset, len);
+
+ memcpy(buf, value + offset, len);
+
+ return len;
+}
+
+struct gatts_incl {
+ u16_t start_handle;
+ u16_t end_handle;
+ u16_t uuid16;
+} __packed;
+
+ssize_t bt_mesh_gatts_attr_read_included(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ void *buf, u16_t len, u16_t offset)
+{
+ struct bt_gatt_attr *incl = attr->user_data;
+ struct bt_uuid *uuid = incl->user_data;
+ struct gatts_incl pdu;
+ u8_t value_len;
+
+ /* first attr points to the start handle */
+ pdu.start_handle = sys_cpu_to_le16(incl->handle);
+ value_len = sizeof(pdu.start_handle) + sizeof(pdu.end_handle);
+
+ /*
+ * Core 4.2, Vol 3, Part G, 3.2,
+ * The Service UUID shall only be present when the UUID is a
+ * 16-bit Bluetooth UUID.
+ */
+ if (uuid->type == BT_UUID_TYPE_16) {
+ pdu.uuid16 = sys_cpu_to_le16(BT_UUID_16(uuid)->val);
+ value_len += sizeof(pdu.uuid16);
+ }
+
+ /* Lookup for service end handle */
+ //bt_mesh_gatts_foreach_attr(incl->handle + 1, 0xffff, get_service_handles, &pdu);
+
+ return bt_mesh_gatts_attr_read(conn, attr, buf, len, offset, &pdu, value_len);
+}
+
+ssize_t bt_mesh_gatts_attr_read_service(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ void *buf, u16_t len, u16_t offset)
+{
+ struct bt_uuid *uuid = attr->user_data;
+
+ if (uuid->type == BT_UUID_TYPE_16) {
+ u16_t uuid16 = sys_cpu_to_le16(BT_UUID_16(uuid)->val);
+
+ return bt_mesh_gatts_attr_read(conn, attr, buf, len, offset, &uuid16, 2);
+ }
+
+ return bt_mesh_gatts_attr_read(conn, attr, buf, len, offset,
+ BT_UUID_128(uuid)->val, 16);
+}
+
+struct gatts_chrc {
+ u8_t properties;
+ u16_t value_handle;
+ union {
+ u16_t uuid16;
+ u8_t uuid[16];
+ };
+} __packed;
+
+ssize_t bt_mesh_gatts_attr_read_chrc(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr, void *buf,
+ u16_t len, u16_t offset)
+{
+ struct bt_gatt_chrc *chrc = attr->user_data;
+ struct gatts_chrc pdu;
+ const struct bt_gatt_attr *next;
+ u8_t value_len;
+
+ pdu.properties = chrc->properties;
+ /* BLUETOOTH SPECIFICATION Version 4.2 [Vol 3, Part G] page 534:
+ * 3.3.2 Characteristic Value Declaration
+ * The Characteristic Value declaration contains the value of the
+ * characteristic. It is the first Attribute after the characteristic
+ * declaration. All characteristic definitions shall have a
+ * Characteristic Value declaration.
+ */
+ next = bt_mesh_gatts_attr_next(attr);
+ if (!next) {
+ BT_WARN("No value for characteristic at 0x%04x", attr->handle);
+ pdu.value_handle = 0x0000;
+ } else {
+ pdu.value_handle = sys_cpu_to_le16(next->handle);
+ }
+ value_len = sizeof(pdu.properties) + sizeof(pdu.value_handle);
+
+ if (chrc->uuid->type == BT_UUID_TYPE_16) {
+ pdu.uuid16 = sys_cpu_to_le16(BT_UUID_16(chrc->uuid)->val);
+ value_len += 2;
+ } else {
+ memcpy(pdu.uuid, BT_UUID_128(chrc->uuid)->val, 16);
+ value_len += 16;
+ }
+
+ return bt_mesh_gatts_attr_read(conn, attr, buf, len, offset, &pdu, value_len);
+}
+
+static void bta_uuid_to_mesh_uuid(tBT_UUID *bta_uuid, const struct bt_uuid *uuid)
+{
+ assert(uuid != NULL && bta_uuid != NULL);
+
+ if (uuid->type == BT_UUID_TYPE_16) {
+ bta_uuid->len = LEN_UUID_16;
+ bta_uuid->uu.uuid16 = BT_UUID_16(uuid)->val;
+ } else if (uuid->type == BT_UUID_TYPE_32) {
+ bta_uuid->len = LEN_UUID_32;
+ bta_uuid->uu.uuid32 = BT_UUID_32(uuid)->val;
+ } else if (uuid->type == BT_UUID_TYPE_128) {
+ bta_uuid->len = LEN_UUID_128;
+ memcpy(bta_uuid->uu.uuid128, BT_UUID_128(uuid)->val, LEN_UUID_128);
+ } else {
+ BT_ERR("%s, Invalid mesh uuid type = %d", __func__, uuid->type);
+ }
+
+ return;
+}
+
+static int gatts_register(struct bt_gatt_service *svc)
+{
+ struct bt_gatt_service *last;
+ u16_t handle;
+ //struct bt_gatt_attr *attrs = svc->attrs;
+ //u16_t count = svc->attr_count;
+
+ if (sys_slist_is_empty(&bt_mesh_gatts_db)) {
+ handle = 0;
+ goto populate;
+ }
+
+ last = SYS_SLIST_PEEK_TAIL_CONTAINER(&bt_mesh_gatts_db, last, node);
+ handle = last->attrs[last->attr_count - 1].handle;
+ BT_DBG("%s, handle = %d", __func__, handle);
+
+populate:
+#if 0
+ /* Populate the handles and append them to the list */
+ for (; attrs && count; attrs++, count--) {
+ if (!attrs->handle) {
+ /* Allocate handle if not set already */
+ attrs->handle = ++handle;
+ } else if (attrs->handle > handle) {
+ /* Use existing handle if valid */
+ handle = attrs->handle;
+ } else {
+ /* Service has conflicting handles */
+ BT_ERR("Unable to register handle 0x%04x",
+ attrs->handle);
+ return -EINVAL;
+ }
+
+ BT_DBG("attr %p handle 0x%04x uuid %s perm 0x%02x",
+ attrs, attrs->handle, bt_uuid_str(attrs->uuid),
+ attrs->perm);
+ }
+#endif
+ sys_slist_append(&bt_mesh_gatts_db, &svc->node);
+ return 0;
+}
+
+static tBTA_GATT_PERM mesh_perm_to_bta_perm(u8_t perm)
+{
+ tBTA_GATT_PERM bta_perm = 0;
+ if ((perm & BT_GATT_PERM_READ) == BT_GATT_PERM_READ) {
+ bta_perm |= BTA_GATT_PERM_READ;
+ }
+
+ if ((perm & BT_GATT_PERM_WRITE) == BT_GATT_PERM_WRITE) {
+ bta_perm |= BTA_GATT_PERM_WRITE;
+ }
+
+ if ((perm & BT_GATT_PERM_READ_ENCRYPT) == BT_GATT_PERM_READ_ENCRYPT) {
+ bta_perm |= BTA_GATT_PERM_READ_ENCRYPTED;
+ }
+
+ if ((perm & BT_GATT_PERM_WRITE_ENCRYPT) == BT_GATT_PERM_WRITE_ENCRYPT) {
+ bta_perm |= BTA_GATT_PERM_WRITE_ENCRYPTED;
+ }
+
+ if ((perm & BT_GATT_PERM_READ_AUTHEN) == BT_GATT_PERM_READ_AUTHEN) {
+ bta_perm |= BTA_GATT_PERM_READ_ENC_MITM;
+ }
+
+ if ((perm & BT_GATT_PERM_WRITE_AUTHEN) == BT_GATT_PERM_WRITE_AUTHEN) {
+ bta_perm |= BTA_GATT_PERM_WRITE_ENC_MITM;
+ }
+
+ return bta_perm;
+}
+
+int bt_mesh_gatts_service_register(struct bt_gatt_service *svc)
+{
+ assert(svc != NULL);
+ tBT_UUID bta_uuid = {0};
+
+ for (int i = 0; i < svc->attr_count; i++) {
+ if (svc->attrs[i].uuid->type == BT_UUID_TYPE_16) {
+ switch (BT_UUID_16(svc->attrs[i].uuid)->val) {
+ case BT_UUID_GATT_PRIMARY_VAL: {
+ future_mesh = future_new();
+ bta_uuid_to_mesh_uuid(&bta_uuid, (struct bt_uuid *)svc->attrs[i].user_data);
+ BTA_GATTS_CreateService(bt_mesh_gatts_if,
+ &bta_uuid, 0, svc->attr_count, true);
+ if (future_await(future_mesh) == FUTURE_FAIL) {
+ BT_ERR("add primary service failed.");
+ return ESP_FAIL;
+ }
+ svc->attrs[i].handle = svc_handle;
+ BT_DBG("############## create service ############");
+ BT_DBG("add pri service: svc_uuid = %x, perm = %d, svc_handle = %d", bta_uuid.uu.uuid16, svc->attrs[i].perm, svc_handle);
+ break;
+ }
+ case BT_UUID_GATT_SECONDARY_VAL: {
+ future_mesh = future_new();
+ bta_uuid_to_mesh_uuid(&bta_uuid, (struct bt_uuid *)svc->attrs[i].user_data);
+ BTA_GATTS_CreateService(bt_mesh_gatts_if,
+ &bta_uuid, 0, svc->attr_count, false);
+ if (future_await(future_mesh) == FUTURE_FAIL) {
+ BT_ERR("add secondary service failed.");
+ return ESP_FAIL;
+ }
+ svc->attrs[i].handle = svc_handle;
+ BT_DBG("add sec service: svc_uuid = %x, perm = %d, svc_handle = %d", bta_uuid.uu.uuid16, svc->attrs[i].perm, svc_handle);
+ break;
+ }
+ case BT_UUID_GATT_INCLUDE_VAL: {
+ break;
+ }
+ case BT_UUID_GATT_CHRC_VAL: {
+ future_mesh = future_new();
+ struct bt_gatt_chrc *gatts_chrc = (struct bt_gatt_chrc *)svc->attrs[i].user_data;
+ bta_uuid_to_mesh_uuid(&bta_uuid, gatts_chrc->uuid);
+ BTA_GATTS_AddCharacteristic(svc_handle, &bta_uuid, mesh_perm_to_bta_perm(svc->attrs[i + 1].perm), gatts_chrc->properties, NULL, NULL);
+ if (future_await(future_mesh) == FUTURE_FAIL) {
+ BT_ERR("Add characristic failed.");
+ return ESP_FAIL;
+ }
+ /* All the characristic should have two handle: the declaration handle and the value handle */
+ svc->attrs[i].handle = char_handle - 1;
+ svc->attrs[i + 1].handle = char_handle;
+ BT_DBG("add char: char_uuid = %x, char_handle = %d, perm = %d, char_pro = %d", BT_UUID_16(gatts_chrc->uuid)->val, char_handle, svc->attrs[i + 1].perm, gatts_chrc->properties);
+ break;
+ }
+ case BT_UUID_GATT_CEP_VAL:
+ case BT_UUID_GATT_CUD_VAL:
+ case BT_UUID_GATT_CCC_VAL:
+ case BT_UUID_GATT_SCC_VAL:
+ case BT_UUID_GATT_CPF_VAL:
+ case BT_UUID_VALID_RANGE_VAL:
+ case BT_UUID_HIDS_EXT_REPORT_VAL:
+ case BT_UUID_HIDS_REPORT_REF_VAL:
+ case BT_UUID_ES_CONFIGURATION_VAL:
+ case BT_UUID_ES_MEASUREMENT_VAL:
+ case BT_UUID_ES_TRIGGER_SETTING_VAL: {
+ future_mesh = future_new();
+ bta_uuid_to_mesh_uuid(&bta_uuid, svc->attrs[i].uuid);
+ BTA_GATTS_AddCharDescriptor(svc_handle, mesh_perm_to_bta_perm(svc->attrs[i].perm), &bta_uuid, NULL, NULL);
+ if (future_await(future_mesh) == FUTURE_FAIL) {
+ BT_ERR("add primary service failed.");
+ return ESP_FAIL;
+ }
+ svc->attrs[i].handle = char_handle;
+ BT_DBG("add descr: descr_uuid = %x, perm= %d, descr_handle = %d", BT_UUID_16(svc->attrs[i].uuid)->val, svc->attrs[i].perm, char_handle);
+ break;
+ }
+ default:
+ break;
+ }
+ }
+ }
+
+ if (svc_handle != 0) {
+ /* TODO: Currently we start service according to function like
+ * bt_mesh_proxy_gatt_enable, etc
+ */
+ //BTA_GATTS_StartService(svc_handle, BTA_GATT_TRANSPORT_LE);
+ svc_handle = 0;
+ }
+
+ // Still should regitster to the adapt bt_mesh_gatts_db.
+ gatts_register(svc);
+ return 0;
+}
+
+int bt_mesh_gatts_disconnect(struct bt_conn *conn, u8_t reason)
+{
+ UNUSED(reason);
+ u16_t conn_id = BT_MESH_GATT_CREATE_CONN_ID(bt_mesh_gatts_if, conn->handle);
+ BTA_GATTS_Close(conn_id);
+ return 0;
+}
+
+int bt_mesh_gatts_service_unregister(struct bt_gatt_service *svc)
+{
+ assert(svc != NULL);
+
+ BTA_GATTS_DeleteService(svc->attrs[0].handle);
+ return 0;
+}
+
+int bt_mesh_gatts_notify(struct bt_conn *conn, const struct bt_gatt_attr *attr,
+ const void *data, u16_t len)
+{
+ u16_t conn_id = BT_MESH_GATT_CREATE_CONN_ID(bt_mesh_gatts_if, conn->handle);
+ BTA_GATTS_HandleValueIndication(conn_id, attr->handle, len, (u8_t *)data, false);
+ return 0;
+}
+
+u16_t bt_mesh_gatt_get_mtu(struct bt_conn *conn)
+{
+ return BTA_GATT_GetLocalMTU();
+}
+
+/* APIs added by Espressif */
+int bt_mesh_gatts_service_stop(struct bt_gatt_service *svc)
+{
+ if (!svc) {
+ BT_ERR("%s, svc should not be NULL", __func__);
+ return -EINVAL;
+ }
+
+ BT_DBG("Stop service:%d", svc->attrs[0].handle);
+
+ BTA_GATTS_StopService(svc->attrs[0].handle);
+ return 0;
+}
+
+int bt_mesh_gatts_service_start(struct bt_gatt_service *svc)
+{
+ if (!svc) {
+ BT_ERR("%s, svc should not be NULL", __func__);
+ return -EINVAL;
+ }
+
+ BT_DBG("Start service:%d", svc->attrs[0].handle);
+
+ BTA_GATTS_StartService(svc->attrs[0].handle, BTA_GATT_TRANSPORT_LE);
+ return 0;
+}
+#endif /* defined(CONFIG_BT_MESH_NODE) && CONFIG_BT_MESH_NODE */
+
+#if defined(CONFIG_BT_MESH_PROVISIONER) && CONFIG_BT_MESH_PROVISIONER
+void bt_mesh_gattc_conn_cb_register(struct bt_prov_conn_cb *cb)
+{
+ bt_mesh_gattc_conn_cb = cb;
+}
+
+u16_t bt_mesh_gattc_get_service_uuid(struct bt_conn *conn)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (conn == &bt_mesh_gattc_info[i].conn) {
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(bt_mesh_gattc_info)) {
+ return 0;
+ }
+
+ return bt_mesh_gattc_info[i].service_uuid;
+}
+
+/** For provisioner acting as a GATT client, it may follow the procedures
+ * listed below.
+ * 1. Create connection with the unprovisioned device
+ * 2. Exchange MTU size
+ * 3. Find Mesh Prov Service in the device's service database
+ * 4. Find Mesh Prov Data In/Out characteristic within the service
+ * 5. Get CCC of Mesh Prov Data Out Characteristic
+ * 6. Set the Notification bit of CCC
+ */
+
+int bt_mesh_gattc_conn_create(const bt_addr_le_t *addr, u16_t service_uuid)
+{
+ u8_t zero[6] = {0};
+ int i;
+
+ if (!addr || !memcmp(addr->a.val, zero, BD_ADDR_LEN) ||
+ (addr->type > BLE_ADDR_RANDOM)) {
+ BT_ERR("%s: invalid address", __func__);
+ return -EINVAL;
+ }
+
+ if (service_uuid != BT_UUID_MESH_PROV_VAL &&
+ service_uuid != BT_UUID_MESH_PROXY_VAL) {
+ BT_ERR("%s: invalid service uuid 0x%04x", __func__, service_uuid);
+ return -EINVAL;
+ }
+
+ /* Check if already creating connection with the device */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (!memcmp(bt_mesh_gattc_info[i].addr, addr->a.val, BD_ADDR_LEN)) {
+ BT_WARN("%s, Already create conn with %s",
+ __func__, bt_hex(addr->a.val, BD_ADDR_LEN));
+ return -EALREADY;
+ }
+ }
+
+ /* Find empty element in queue to store device info */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if ((bt_mesh_gattc_info[i].conn.handle == 0xFFFF) &&
+ (bt_mesh_gattc_info[i].service_uuid == 0x0000)) {
+ memcpy(bt_mesh_gattc_info[i].addr, addr->a.val, BD_ADDR_LEN);
+ bt_mesh_gattc_info[i].addr_type = addr->type;
+ /* Service to be found after exhanging mtu size */
+ bt_mesh_gattc_info[i].service_uuid = service_uuid;
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(bt_mesh_gattc_info)) {
+ BT_WARN("%s, gattc info is full", __func__);
+ return -ENOMEM;
+ }
+
+#if BT_DEV
+ if (atomic_test_and_clear_bit(bt_dev.flags, BT_DEV_EXPLICIT_SCAN)) {
+ BTM_BleScan(false, 0, NULL, NULL, NULL);
+ }
+#else
+ BTM_BleScan(false, 0, NULL, NULL, NULL);
+#endif /* BT_DEV */
+
+ BT_DBG("%s, create conn with %s", __func__, bt_hex(addr->a.val, BD_ADDR_LEN));
+
+ /* Min_interval: 250ms
+ * Max_interval: 250ms
+ * Slave_latency: 0x0
+ * Supervision_timeout: 32 sec
+ */
+ BTA_DmSetBlePrefConnParams(bt_mesh_gattc_info[i].addr, 0xC8, 0xC8, 0x00, 0xC80);
+
+ BTA_GATTC_Open(bt_mesh_gattc_if, bt_mesh_gattc_info[i].addr,
+ bt_mesh_gattc_info[i].addr_type, true, BTA_GATT_TRANSPORT_LE);
+
+ /* Increment pbg_count */
+ provisioner_pbg_count_inc();
+
+ return 0;
+}
+
+void bt_mesh_gattc_exchange_mtu(u8_t index)
+{
+ /** Set local MTU and exchange with GATT server.
+ * ATT_MTU >= 69 for Mesh GATT Prov Service
+ * ATT_NTU >= 33 for Mesh GATT Proxy Service
+ */
+ u16_t conn_id;
+
+ conn_id = BT_MESH_GATT_CREATE_CONN_ID(bt_mesh_gattc_if, bt_mesh_gattc_info[index].conn.handle);
+
+ BTA_GATTC_ConfigureMTU(conn_id);
+}
+
+u16_t bt_mesh_gattc_get_mtu_info(struct bt_conn *conn)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (conn == &bt_mesh_gattc_info[i].conn) {
+ return bt_mesh_gattc_info[i].mtu;
+ }
+ }
+
+ return 0;
+}
+
+int bt_mesh_gattc_write_no_rsp(struct bt_conn *conn, const struct bt_gatt_attr *attr,
+ const void *data, u16_t len)
+{
+ int i;
+ u16_t conn_id;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (conn == &bt_mesh_gattc_info[i].conn) {
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(bt_mesh_gattc_info)) {
+ BT_ERR("No conn found");
+ /** Here we return 0 for prov_send() return value check in provisioner.c
+ */
+ return 0;
+ }
+
+ conn_id = BT_MESH_GATT_CREATE_CONN_ID(bt_mesh_gattc_if, bt_mesh_gattc_info[i].conn.handle);
+
+ BTA_GATTC_WriteCharValue(conn_id, bt_mesh_gattc_info[i].data_in_handle,
+ BTA_GATTC_TYPE_WRITE_NO_RSP, len,
+ (u8_t *)data, BTA_GATT_AUTH_REQ_NONE);
+
+ return 0;
+}
+
+void bt_mesh_gattc_disconnect(struct bt_conn *conn)
+{
+ /** Disconnect
+ * Clear proper proxy server information
+ * Clear proper prov_link information
+ * Clear proper bt_mesh_gattc_info information
+ * Here in adapter, we just clear proper bt_mesh_gattc_info, and
+ * when proxy_disconnected callback comes, the proxy server
+ * information and prov_link information should be cleared.
+ */
+ int i;
+ u16_t conn_id;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (conn == &bt_mesh_gattc_info[i].conn) {
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(bt_mesh_gattc_info)) {
+ BT_ERR("No conn found");
+ return;
+ }
+
+ conn_id = BT_MESH_GATT_CREATE_CONN_ID(bt_mesh_gattc_if, bt_mesh_gattc_info[i].conn.handle);
+
+ BTA_GATTC_Close(conn_id);
+}
+
+/** Mesh Provisioning Service: 0x1827
+ * Mesh Provisioning Data In: 0x2ADB
+ * Mesh Provisioning Data Out: 0x2ADC
+ * Mesh Proxy Service: 0x1828
+ * Mesh Proxy Data In: 0x2ADD
+ * Mesh PROXY Data Out: 0x2ADE
+ */
+static void bt_mesh_bta_gattc_cb(tBTA_GATTC_EVT event, tBTA_GATTC *p_data)
+{
+ struct bt_conn *conn = NULL;
+ u16_t handle = 0;
+ ssize_t len = 0;
+ int i = 0;
+
+ switch (event) {
+ case BTA_GATTC_REG_EVT:
+ if (p_data->reg_oper.status == BTA_GATT_OK) {
+ u8_t uuid[16] = { [0 ... 15] = BT_MESH_GATTC_APP_UUID_BYTE };
+
+ BT_DBG("BTA_GATTC_REG_EVT");
+
+ if (p_data->reg_oper.app_uuid.len == LEN_UUID_128 &&
+ !memcmp(p_data->reg_oper.app_uuid.uu.uuid128, uuid, 16)) {
+ bt_mesh_gattc_if = p_data->reg_oper.client_if;
+ BT_DBG("bt_mesh_gattc_if is %d", bt_mesh_gattc_if);
+ }
+ }
+ break;
+ case BTA_GATTC_CFG_MTU_EVT: {
+ if (p_data->cfg_mtu.status == BTA_GATT_OK) {
+ BT_DBG("BTA_GATTC_CFG_MTU_EVT, cfg_mtu is %d", p_data->cfg_mtu.mtu);
+
+ handle = BT_MESH_GATT_GET_CONN_ID(p_data->cfg_mtu.conn_id);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (bt_mesh_gattc_info[i].conn.handle == handle) {
+ bt_mesh_gattc_info[i].mtu = p_data->cfg_mtu.mtu;
+ break;
+ }
+ }
+
+ /** Once mtu exchanged accomplished, start to find services, and here
+ * need a flag to indicate which service to find(Mesh Prov Service or
+ * Mesh Proxy Service)
+ */
+ if (i != ARRAY_SIZE(bt_mesh_gattc_info)) {
+ tBT_UUID service_uuid;
+ u16_t conn_id;
+
+ conn_id = BT_MESH_GATT_CREATE_CONN_ID(bt_mesh_gattc_if, bt_mesh_gattc_info[i].conn.handle);
+ service_uuid.len = sizeof(bt_mesh_gattc_info[i].service_uuid);
+ service_uuid.uu.uuid16 = bt_mesh_gattc_info[i].service_uuid;
+
+ /* Search Mesh Provisioning Service or Mesh Proxy Service */
+ BTA_GATTC_ServiceSearchRequest(conn_id, &service_uuid);
+ }
+ }
+ break;
+ }
+ case BTA_GATTC_SEARCH_RES_EVT: {
+ BT_DBG("BTA_GATTC_SEARCH_RES_EVT");
+
+ handle = BT_MESH_GATT_GET_CONN_ID(p_data->srvc_res.conn_id);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (bt_mesh_gattc_info[i].conn.handle == handle) {
+ break;
+ }
+ }
+
+ if (i != ARRAY_SIZE(bt_mesh_gattc_info)) {
+ if (p_data->srvc_res.service_uuid.uuid.len == 2 &&
+ p_data->srvc_res.service_uuid.uuid.uu.uuid16 == bt_mesh_gattc_info[i].service_uuid) {
+ bt_mesh_gattc_info[i].start_handle = p_data->srvc_res.start_handle;
+ bt_mesh_gattc_info[i].end_handle = p_data->srvc_res.end_handle;
+ }
+ }
+ break;
+ }
+ case BTA_GATTC_SEARCH_CMPL_EVT: {
+ if (p_data->search_cmpl.status == BTA_GATT_OK) {
+ BT_DBG("BTA_GATTC_SEARCH_CMPL_EVT");
+
+ handle = BT_MESH_GATT_GET_CONN_ID(p_data->search_cmpl.conn_id);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (bt_mesh_gattc_info[i].conn.handle == handle) {
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(bt_mesh_gattc_info)) {
+ BT_ERR("BTA_GATTC_SEARCH_CMPL_EVT: conn_id not found");
+ return;
+ }
+
+ conn = &bt_mesh_gattc_info[i].conn;
+
+ if (bt_mesh_gattc_info[i].start_handle == 0x00 ||
+ bt_mesh_gattc_info[i].end_handle == 0x00 ||
+ (bt_mesh_gattc_info[i].start_handle > bt_mesh_gattc_info[i].end_handle)) {
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ int count = 0;
+ int num = 0;
+ u16_t conn_id;
+ tBT_UUID char_uuid;
+ btgatt_db_element_t *result = NULL;
+ tBTA_GATT_STATUS status;
+ u16_t notify_en = BT_GATT_CCC_NOTIFY;
+ tBTA_GATT_UNFMT write;
+
+ /* Get the characteristic num within Mesh Provisioning/Proxy Service */
+ conn_id = BT_MESH_GATT_CREATE_CONN_ID(bt_mesh_gattc_if, bt_mesh_gattc_info[i].conn.handle);
+ BTA_GATTC_GetDBSizeByType(conn_id, BTGATT_DB_CHARACTERISTIC, bt_mesh_gattc_info[i].start_handle,
+ bt_mesh_gattc_info[i].end_handle, BTA_GATTC_INVALID_HANDLE, &count);
+ if (count != 2) {
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ /* Get Mesh Provisioning/Proxy Data In/Out Characteristic */
+ for (int j = 0; j != 2; j++) {
+ /** First: find Mesh Provisioning/Proxy Data In Characteristic
+ * Second: find Mesh Provisioning/Proxy Data Out Characteristic
+ */
+ char_uuid.len = 2;
+ if (bt_mesh_gattc_info[i].service_uuid == BT_UUID_MESH_PROV_VAL) {
+ char_uuid.uu.uuid16 = BT_UUID_MESH_PROV_DATA_IN_VAL + j;
+ } else if (bt_mesh_gattc_info[i].service_uuid == BT_UUID_MESH_PROXY_VAL) {
+ char_uuid.uu.uuid16 = BT_UUID_MESH_PROXY_DATA_IN_VAL + j;
+ }
+
+ BTA_GATTC_GetCharByUUID(conn_id, bt_mesh_gattc_info[i].start_handle,
+ bt_mesh_gattc_info[i].end_handle, char_uuid, &result, &num);
+
+ if (!result) {
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ if (num != 1) {
+ osi_free(result);
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ if (!j) {
+ if (!(result[0].properties & BT_GATT_CHRC_WRITE_WITHOUT_RESP)) {
+ osi_free(result);
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+ bt_mesh_gattc_info[i].data_in_handle = result[0].attribute_handle;
+ } else {
+ if (!(result[0].properties & BT_GATT_CHRC_NOTIFY)) {
+ osi_free(result);
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+ bt_mesh_gattc_info[i].data_out_handle = result[0].attribute_handle;
+ }
+ osi_free(result);
+ result = NULL;
+ }
+
+ /* Register Notification fot Mesh Provisioning/Proxy Data Out Characteristic */
+ status = BTA_GATTC_RegisterForNotifications(bt_mesh_gattc_if, bt_mesh_gattc_info[i].addr,
+ bt_mesh_gattc_info[i].data_out_handle);
+ if (status != BTA_GATT_OK) {
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ /** After notification is registered, get descriptor number of the
+ * Mesh Provisioning/Proxy Data Out Characteristic
+ */
+ BTA_GATTC_GetDBSizeByType(conn_id, BTGATT_DB_DESCRIPTOR, bt_mesh_gattc_info[i].start_handle,
+ bt_mesh_gattc_info[i].end_handle, bt_mesh_gattc_info[i].data_out_handle, &num);
+ if (!num) {
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ /* Get CCC of Mesh Provisioning/Proxy Data Out Characteristic */
+ char_uuid.len = 2;
+ char_uuid.uu.uuid16 = BT_UUID_GATT_CCC_VAL;
+ BTA_GATTC_GetDescrByCharHandle(conn_id, bt_mesh_gattc_info[i].data_out_handle,
+ char_uuid, &result, &num);
+
+ if (!result) {
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ if (num != 1) {
+ osi_free(result);
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ bt_mesh_gattc_info[i].ccc_handle = result[0].attribute_handle;
+
+ /** Enable Notification of Mesh Provisioning/Proxy Data Out
+ * Characteristic Descriptor.
+ */
+ write.len = sizeof(notify_en);
+ write.p_value = (u8_t *)¬ify_en;
+ BTA_GATTC_WriteCharDescr(conn_id, result[0].attribute_handle,
+ BTA_GATTC_TYPE_WRITE, &write, BTA_GATT_AUTH_REQ_NONE);
+
+ osi_free(result);
+ result = NULL;
+ }
+ break;
+ }
+ case BTA_GATTC_READ_DESCR_EVT:
+ break;
+ case BTA_GATTC_WRITE_DESCR_EVT: {
+ if (p_data->write.status == BTA_GATT_OK) {
+ BT_DBG("BTA_GATTC_WRITE_DESCR_EVT");
+
+ handle = BT_MESH_GATT_GET_CONN_ID(p_data->write.conn_id);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (bt_mesh_gattc_info[i].conn.handle == handle) {
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(bt_mesh_gattc_info)) {
+ BT_ERR("BTA_GATTC_WRITE_DESCR_EVT: conn_id not found");
+ return;
+ }
+
+ conn = &bt_mesh_gattc_info[i].conn;
+
+ if (bt_mesh_gattc_info[i].ccc_handle != p_data->write.handle) {
+ BT_WARN("BTA_GATTC_WRITE_DESCR_EVT: ccc_handle not match");
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ if (bt_mesh_gattc_info[i].service_uuid == BT_UUID_MESH_PROV_VAL) {
+ if (bt_mesh_gattc_conn_cb != NULL && bt_mesh_gattc_conn_cb->prov_write_descr != NULL) {
+ len = bt_mesh_gattc_conn_cb->prov_write_descr(&bt_mesh_gattc_info[i].conn, bt_mesh_gattc_info[i].addr);
+ if (len < 0) {
+ BT_ERR("prov_write_descr fail");
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+ bt_mesh_gattc_info[i].wr_desc_done = true;
+ }
+ } else if (bt_mesh_gattc_info[i].service_uuid == BT_UUID_MESH_PROXY_VAL) {
+ if (bt_mesh_gattc_conn_cb != NULL && bt_mesh_gattc_conn_cb->proxy_write_descr != NULL) {
+ len = bt_mesh_gattc_conn_cb->proxy_write_descr(&bt_mesh_gattc_info[i].conn);
+ if (len < 0) {
+ BT_ERR("proxy_write_descr fail");
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+ }
+ }
+ }
+ break;
+ }
+ case BTA_GATTC_NOTIF_EVT: {
+ BT_DBG("BTA_GATTC_NOTIF_EVT");
+
+ handle = BT_MESH_GATT_GET_CONN_ID(p_data->notify.conn_id);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (bt_mesh_gattc_info[i].conn.handle == handle) {
+ break;
+ }
+ }
+
+ if (i == ARRAY_SIZE(bt_mesh_gattc_info)) {
+ BT_ERR("BTA_GATTC_WRITE_DESCR_EVT: conn_id not found");
+ return;
+ }
+
+ conn = &bt_mesh_gattc_info[i].conn;
+
+ if (memcmp(bt_mesh_gattc_info[i].addr, p_data->notify.bda, BD_ADDR_LEN) ||
+ bt_mesh_gattc_info[i].data_out_handle != p_data->notify.handle ||
+ p_data->notify.is_notify == false) {
+ BT_ERR("Notification error");
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+
+ if (bt_mesh_gattc_info[i].service_uuid == BT_UUID_MESH_PROV_VAL) {
+ if (bt_mesh_gattc_conn_cb != NULL && bt_mesh_gattc_conn_cb->prov_notify != NULL) {
+ len = bt_mesh_gattc_conn_cb->prov_notify(&bt_mesh_gattc_info[i].conn,
+ p_data->notify.value, p_data->notify.len);
+ if (len < 0) {
+ BT_ERR("Receive prov_notify fail");
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+ }
+ } else if (bt_mesh_gattc_info[i].service_uuid == BT_UUID_MESH_PROXY_VAL) {
+ if (bt_mesh_gattc_conn_cb != NULL && bt_mesh_gattc_conn_cb->proxy_notify != NULL) {
+ len = bt_mesh_gattc_conn_cb->proxy_notify(&bt_mesh_gattc_info[i].conn,
+ p_data->notify.value, p_data->notify.len);
+ if (len < 0) {
+ BT_ERR("Receive proxy_notify fail");
+ bt_mesh_gattc_disconnect(conn);
+ return;
+ }
+ }
+ }
+ break;
+ }
+ case BTA_GATTC_READ_CHAR_EVT:
+ break;
+ case BTA_GATTC_WRITE_CHAR_EVT:
+ break;
+ case BTA_GATTC_PREP_WRITE_EVT:
+ break;
+ case BTA_GATTC_EXEC_EVT:
+ break;
+ case BTA_GATTC_OPEN_EVT: {
+ BT_DBG("BTA_GATTC_OPEN_EVT");
+ /** After current connection is established, provisioner can
+ * use BTA_DmBleScan() to re-enable scan.
+ */
+ tBTM_STATUS status;
+#if BT_DEV
+ if (!atomic_test_and_set_bit(bt_dev.flags, BT_DEV_EXPLICIT_SCAN)) {
+ status = BTM_BleScan(true, 0, ble_mesh_scan_results_cb, NULL, NULL);
+ if (status != BTM_SUCCESS && status != BTM_CMD_STARTED) {
+ BT_ERR("%s, invalid status %d", __func__, status);
+ break;
+ }
+ }
+#else
+ status = BTM_BleScan(true, 0, ble_mesh_scan_results_cb, NULL, NULL);
+ if (status != BTM_SUCCESS && status != BTM_CMD_STARTED) {
+ BT_ERR("%s, invalid status %d", __func__, status);
+ break;
+ }
+#endif /* BT_DEV */
+ break;
+ }
+ case BTA_GATTC_CLOSE_EVT:
+ BT_DBG("BTA_GATTC_CLOSE_EVT");
+ break;
+ case BTA_GATTC_CONNECT_EVT: {
+ BT_DBG("BTA_GATTC_CONNECT_EVT");
+
+ if (bt_mesh_gattc_if != p_data->connect.client_if) {
+ BT_ERR("bt_mesh_gattc_if & connect_if don't match");
+ return;
+ }
+
+ if (bt_mesh_gattc_conn_cb != NULL && bt_mesh_gattc_conn_cb->connected != NULL) {
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (!memcmp(bt_mesh_gattc_info[i].addr, p_data->connect.remote_bda, BD_ADDR_LEN)) {
+ bt_mesh_gattc_info[i].conn.handle = BT_MESH_GATT_GET_CONN_ID(p_data->connect.conn_id);
+ (bt_mesh_gattc_conn_cb->connected)(bt_mesh_gattc_info[i].addr, &bt_mesh_gattc_info[i].conn, i);
+ break;
+ }
+ }
+ }
+ break;
+ }
+ case BTA_GATTC_DISCONNECT_EVT: {
+ BT_DBG("BTA_GATTC_DISCONNECT_EVT");
+
+ if (bt_mesh_gattc_if != p_data->disconnect.client_if) {
+ BT_ERR("bt_mesh_gattc_if & disconnect_if don't match");
+ return;
+ }
+
+ handle = BT_MESH_GATT_GET_CONN_ID(p_data->disconnect.conn_id);
+
+ if (bt_mesh_gattc_conn_cb != NULL && bt_mesh_gattc_conn_cb->disconnected != NULL) {
+ for (i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ if (!memcmp(bt_mesh_gattc_info[i].addr, p_data->disconnect.remote_bda, BD_ADDR_LEN)) {
+ if (bt_mesh_gattc_info[i].conn.handle == handle) {
+ (bt_mesh_gattc_conn_cb->disconnected)(&bt_mesh_gattc_info[i].conn, p_data->disconnect.reason);
+ if (!bt_mesh_gattc_info[i].wr_desc_done) {
+ /* Add this in case connection is established, connected event comes, but
+ * connection is terminated before server->filter_type is set to PROV.
+ */
+ provisioner_clear_link_conn_info(bt_mesh_gattc_info[i].addr);
+ }
+ } else {
+ /* Add this in case connection is failed to be established, and here we
+ * need to clear some provision link info, like connecting flag, device
+ * uuid, address info, etc.
+ */
+ provisioner_clear_link_conn_info(bt_mesh_gattc_info[i].addr);
+ }
+ /* Decrease prov pbg_count */
+ provisioner_pbg_count_dec();
+ /* Reset corresponding gattc info */
+ memset(&bt_mesh_gattc_info[i], 0, sizeof(bt_mesh_gattc_info[i]));
+ bt_mesh_gattc_info[i].conn.handle = 0xFFFF;
+ bt_mesh_gattc_info[i].mtu = GATT_DEF_BLE_MTU_SIZE;
+ bt_mesh_gattc_info[i].wr_desc_done = false;
+ break;
+ }
+ }
+ }
+ break;
+ }
+ case BTA_GATTC_CONGEST_EVT:
+ break;
+ case BTA_GATTC_SRVC_CHG_EVT:
+ break;
+ default:
+ break;
+ }
+}
+#endif /* defined(CONFIG_BT_MESH_PROVISIONER) && CONFIG_BT_MESH_PROVISIONER */
+
+struct bt_conn *bt_mesh_conn_ref(struct bt_conn *conn)
+{
+ atomic_inc(&conn->ref);
+
+ BT_DBG("handle %u ref %u", conn->handle, atomic_get(&conn->ref));
+
+ return conn;
+}
+
+void bt_mesh_conn_unref(struct bt_conn *conn)
+{
+ atomic_dec(&conn->ref);
+
+ BT_DBG("handle %u ref %u", conn->handle, atomic_get(&conn->ref));
+}
+
+void bt_mesh_gatt_init(void)
+{
+ tBT_UUID app_uuid = {LEN_UUID_128, {0}};
+
+ BTA_GATT_SetLocalMTU(GATT_DEF_BLE_MTU_SIZE);
+
+#if CONFIG_BT_MESH_NODE
+ /* Fill our internal UUID with a fixed pattern 0x96 for the ble mesh */
+ memset(&app_uuid.uu.uuid128, 0x96, LEN_UUID_128);
+ BTA_GATTS_AppRegister(&app_uuid, bt_mesh_bta_gatts_cb);
+#endif
+
+#if CONFIG_BT_MESH_PROVISIONER
+ for (int i = 0; i < ARRAY_SIZE(bt_mesh_gattc_info); i++) {
+ bt_mesh_gattc_info[i].conn.handle = 0xFFFF;
+ bt_mesh_gattc_info[i].mtu = GATT_DEF_BLE_MTU_SIZE; /* Default MTU_SIZE 23 */
+ bt_mesh_gattc_info[i].wr_desc_done = false;
+ }
+ memset(&app_uuid.uu.uuid128, BT_MESH_GATTC_APP_UUID_BYTE, LEN_UUID_128);
+ BTA_GATTC_AppRegister(&app_uuid, bt_mesh_bta_gattc_cb);
+#endif
+}
+
+void bt_mesh_adapt_init(void)
+{
+ BT_DBG("%s", __func__);
+ /* initialization of P-256 parameters */
+ p_256_init_curve(KEY_LENGTH_DWORDS_P256);
+}
+
+int bt_mesh_rand(void *buf, size_t len)
+{
+ if (!len) {
+ return -EAGAIN;
+ }
+
+ // Reset the buf value to the fixed value.
+ memset(buf, 0x55, len);
+
+ for (int i = 0; i < (int)(len / sizeof(u32_t)); i++) {
+ u32_t rand = esp_random();
+ memcpy(buf + i * sizeof(u32_t), &rand, sizeof(u32_t));
+ }
+
+ BT_DBG("%s, rand: %s", __func__, bt_hex(buf, len));
+ return 0;
+}
+
+void bt_mesh_set_private_key(const u8_t pri_key[32])
+{
+ memcpy(bt_mesh_private_key, pri_key, 32);
+}
+
+const u8_t *bt_mesh_pub_key_get(void)
+{
+ Point public_key;
+ BT_OCTET32 pri_key;
+#if 1
+ if (atomic_test_bit(bt_dev.flags, BT_DEV_HAS_PUB_KEY)) {
+ return bt_mesh_public_key;
+ }
+#else
+ /* BLE Mesh BQB test case MESH/NODE/PROV/UPD/BV-12-C requires
+ * different public key for each provisioning procedure.
+ * Note: if enabled, when Provisioner provision multiple devices
+ * at the same time, this may cause invalid confirmation value.
+ */
+ if (bt_mesh_rand(bt_mesh_private_key, 32)) {
+ BT_ERR("Unable to generate bt_mesh_private_key");
+ return NULL;
+ }
+#endif
+ mem_rcopy(pri_key, bt_mesh_private_key, 32);
+ ECC_PointMult(&public_key, &(curve_p256.G), (DWORD *)pri_key, KEY_LENGTH_DWORDS_P256);
+
+ memcpy(bt_mesh_public_key, public_key.x, BT_OCTET32_LEN);
+ memcpy(bt_mesh_public_key + BT_OCTET32_LEN, public_key.y, BT_OCTET32_LEN);
+
+ atomic_set_bit(bt_dev.flags, BT_DEV_HAS_PUB_KEY);
+ BT_DBG("gen the bt_mesh_public_key:%s", bt_hex(bt_mesh_public_key, sizeof(bt_mesh_public_key)));
+
+ return bt_mesh_public_key;
+}
+
+bool bt_mesh_check_public_key(const u8_t key[64])
+{
+ struct p256_pub_key {
+ u8_t x[32];
+ u8_t y[32];
+ } check = {0};
+
+ sys_memcpy_swap(check.x, key, 32);
+ sys_memcpy_swap(check.y, key + 32, 32);
+
+ return ECC_CheckPointIsInElliCur_P256((Point *)&check);
+}
+
+int bt_mesh_dh_key_gen(const u8_t remote_pk[64], bt_dh_key_cb_t cb)
+{
+ BT_OCTET32 private_key;
+ Point peer_publ_key;
+ Point new_publ_key;
+ BT_OCTET32 dhkey;
+
+ BT_DBG("private key = %s", bt_hex(bt_mesh_private_key, BT_OCTET32_LEN));
+
+ mem_rcopy(private_key, bt_mesh_private_key, BT_OCTET32_LEN);
+ memcpy(peer_publ_key.x, remote_pk, BT_OCTET32_LEN);
+ memcpy(peer_publ_key.y, &remote_pk[BT_OCTET32_LEN], BT_OCTET32_LEN);
+
+ BT_DBG("remote public key x = %s", bt_hex(peer_publ_key.x, BT_OCTET32_LEN));
+ BT_DBG("remote public key y = %s", bt_hex(peer_publ_key.y, BT_OCTET32_LEN));
+
+ ECC_PointMult(&new_publ_key, &peer_publ_key, (DWORD *) private_key, KEY_LENGTH_DWORDS_P256);
+
+ memcpy(dhkey, new_publ_key.x, BT_OCTET32_LEN);
+
+ BT_DBG("new public key x = %s", bt_hex(new_publ_key.x, 32));
+ BT_DBG("new public key y = %s", bt_hex(new_publ_key.y, 32));
+
+ if (cb != NULL) {
+ cb((const u8_t *)dhkey);
+ }
+
+ return 0;
+}
+
+static void ecb_encrypt(u8_t const *const key_le, u8_t const *const clear_text_le,
+ u8_t *const cipher_text_le, u8_t *const cipher_text_be)
+{
+ struct ecb_param ecb;
+ mbedtls_aes_context aes_ctx = {0};
+
+ aes_ctx.key_bytes = 16;
+ mem_rcopy(&aes_ctx.key[0], key_le, 16);
+ mem_rcopy(&ecb.clear_text[0], clear_text_le, sizeof(ecb.clear_text));
+ mbedtls_aes_crypt_ecb(&aes_ctx, MBEDTLS_AES_ENCRYPT, &ecb.clear_text[0], &ecb.cipher_text[0]);
+
+ if (cipher_text_le) {
+ mem_rcopy(cipher_text_le, &ecb.cipher_text[0],
+ sizeof(ecb.cipher_text));
+ }
+
+ if (cipher_text_be) {
+ memcpy(cipher_text_be, &ecb.cipher_text[0],
+ sizeof(ecb.cipher_text));
+ }
+}
+
+static void ecb_encrypt_be(u8_t const *const key_be, u8_t const *const clear_text_be,
+ u8_t *const cipher_text_be)
+{
+ struct ecb_param ecb;
+ mbedtls_aes_context aes_ctx = {0};
+
+ aes_ctx.key_bytes = 16;
+ memcpy(&aes_ctx.key[0], key_be, 16);
+ memcpy(&ecb.clear_text[0], clear_text_be, sizeof(ecb.clear_text));
+ mbedtls_aes_crypt_ecb(&aes_ctx, MBEDTLS_AES_ENCRYPT, &ecb.clear_text[0], &ecb.cipher_text[0]);
+
+ memcpy(cipher_text_be, &ecb.cipher_text[0], sizeof(ecb.cipher_text));
+}
+
+int bt_encrypt_le(const u8_t key[16], const u8_t plaintext[16],
+ u8_t enc_data[16])
+{
+#if CONFIG_MBEDTLS_HARDWARE_AES
+ BT_DBG("key %s plaintext %s", bt_hex(key, 16), bt_hex(plaintext, 16));
+
+ ecb_encrypt(key, plaintext, enc_data, NULL);
+
+ BT_DBG("enc_data %s", bt_hex(enc_data, 16));
+ return 0;
+#else /* CONFIG_MBEDTLS_HARDWARE_AES */
+ struct tc_aes_key_sched_struct s;
+ u8_t tmp[16];
+
+ BT_DBG("key %s plaintext %s", bt_hex(key, 16), bt_hex(plaintext, 16));
+
+ sys_memcpy_swap(tmp, key, 16);
+
+ if (tc_aes128_set_encrypt_key(&s, tmp) == TC_CRYPTO_FAIL) {
+ return -EINVAL;
+ }
+
+ sys_memcpy_swap(tmp, plaintext, 16);
+
+ if (tc_aes_encrypt(enc_data, tmp, &s) == TC_CRYPTO_FAIL) {
+ return -EINVAL;
+ }
+
+ sys_mem_swap(enc_data, 16);
+
+ BT_DBG("enc_data %s", bt_hex(enc_data, 16));
+
+ return 0;
+#endif /* CONFIG_MBEDTLS_HARDWARE_AES */
+}
+
+int bt_encrypt_be(const u8_t key[16], const u8_t plaintext[16],
+ u8_t enc_data[16])
+{
+#if CONFIG_MBEDTLS_HARDWARE_AES
+ BT_DBG("key %s plaintext %s", bt_hex(key, 16), bt_hex(plaintext, 16));
+
+ ecb_encrypt_be(key, plaintext, enc_data);
+
+ BT_DBG("enc_data %s", bt_hex(enc_data, 16));
+
+ return 0;
+#else /* CONFIG_MBEDTLS_HARDWARE_AES */
+ struct tc_aes_key_sched_struct s;
+
+ BT_DBG("key %s plaintext %s", bt_hex(key, 16), bt_hex(plaintext, 16));
+
+ if (tc_aes128_set_encrypt_key(&s, key) == TC_CRYPTO_FAIL) {
+ return -EINVAL;
+ }
+
+ if (tc_aes_encrypt(enc_data, plaintext, &s) == TC_CRYPTO_FAIL) {
+ return -EINVAL;
+ }
+
+ BT_DBG("enc_data %s", bt_hex(enc_data, 16));
+
+ return 0;
+#endif /* CONFIG_MBEDTLS_HARDWARE_AES */
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/mesh_core/mesh_buf.c b/components/bt/ble_mesh/mesh_core/mesh_buf.c
new file mode 100644
index 0000000000..fe74fc0799
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_buf.c
@@ -0,0 +1,528 @@
+/*
+ * Copyright (c) 2015 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+
+#include "mesh_buf.h"
+#include "mesh_trace.h"
+#include "esp_bt_defs.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+
+/* Helpers to access the storage array, since we don't have access to its
+ * type at this point anymore.
+ */
+#define BUF_SIZE(pool) (sizeof(struct net_buf) + \
+ ROUND_UP(pool->buf_size, 4) + \
+ ROUND_UP(pool->user_data_size, 4))
+#define UNINIT_BUF(pool, n) (struct net_buf *)(((u8_t *)(pool->__bufs)) + \
+ ((n) * BUF_SIZE(pool)))
+
+/* Linker-defined symbol bound to the static pool structs */
+struct net_buf_pool *net_buf_pool_get(int id)
+{
+ return (struct net_buf_pool *)id;
+}
+
+//static int pool_id(struct net_buf_pool *pool)
+//{
+// BT_DBG("%s, the pool id = %p", __func__, pool);
+// return (int)&pool[pool->buf_count - pool->uninit_count - 1];
+//}
+
+static inline struct net_buf *pool_get_uninit(struct net_buf_pool *pool,
+ u16_t uninit_count)
+{
+ struct net_buf *buf;
+
+ buf = UNINIT_BUF(pool, pool->buf_count - uninit_count);
+
+ buf->pool_id = pool;
+ buf->size = pool->buf_size;
+ BT_DBG("%s, pool_id = %p, uninit_count = %d, pool->buf_count = %d, buf = %p, buf->size = %d, pool = %p",
+ __func__, buf->pool_id, uninit_count, pool->buf_count, buf, buf->size, pool);
+
+ return buf;
+}
+
+void *net_buf_simple_add(struct net_buf_simple *buf, size_t len)
+{
+ u8_t *tail = net_buf_simple_tail(buf);
+
+ NET_BUF_SIMPLE_DBG("buf %p len %u", buf, len);
+
+ NET_BUF_SIMPLE_ASSERT(net_buf_simple_tailroom(buf) >= len);
+
+ buf->len += len;
+ return tail;
+}
+
+void *net_buf_simple_add_mem(struct net_buf_simple *buf, const void *mem,
+ size_t len)
+{
+ NET_BUF_SIMPLE_DBG("buf %p len %u", buf, len);
+
+ return memcpy(net_buf_simple_add(buf, len), mem, len);
+}
+
+u8_t *net_buf_simple_add_u8(struct net_buf_simple *buf, u8_t val)
+{
+ u8_t *u8;
+
+ NET_BUF_SIMPLE_DBG("buf %p val 0x%02x", buf, val);
+
+ u8 = net_buf_simple_add(buf, 1);
+ *u8 = val;
+
+ return u8;
+}
+
+void net_buf_simple_add_le16(struct net_buf_simple *buf, u16_t val)
+{
+ NET_BUF_SIMPLE_DBG("buf %p val %u", buf, val);
+
+ val = sys_cpu_to_le16(val);
+ memcpy(net_buf_simple_add(buf, sizeof(val)), &val, sizeof(val));
+}
+
+void net_buf_simple_add_be16(struct net_buf_simple *buf, u16_t val)
+{
+ NET_BUF_SIMPLE_DBG("buf %p val %u", buf, val);
+
+ val = sys_cpu_to_be16(val);
+ memcpy(net_buf_simple_add(buf, sizeof(val)), &val, sizeof(val));
+}
+
+void net_buf_simple_add_le32(struct net_buf_simple *buf, u32_t val)
+{
+ NET_BUF_SIMPLE_DBG("buf %p val %u", buf, val);
+
+ val = sys_cpu_to_le32(val);
+ memcpy(net_buf_simple_add(buf, sizeof(val)), &val, sizeof(val));
+}
+
+void net_buf_simple_add_be32(struct net_buf_simple *buf, u32_t val)
+{
+ NET_BUF_SIMPLE_DBG("buf %p val %u", buf, val);
+
+ val = sys_cpu_to_be32(val);
+ memcpy(net_buf_simple_add(buf, sizeof(val)), &val, sizeof(val));
+}
+
+void *net_buf_simple_push(struct net_buf_simple *buf, size_t len)
+{
+ NET_BUF_SIMPLE_DBG("buf %p len %u", buf, len);
+
+ NET_BUF_SIMPLE_ASSERT(net_buf_simple_headroom(buf) >= len);
+
+ buf->data -= len;
+ buf->len += len;
+ return buf->data;
+}
+
+void net_buf_simple_push_le16(struct net_buf_simple *buf, u16_t val)
+{
+ NET_BUF_SIMPLE_DBG("buf %p val %u", buf, val);
+
+ val = sys_cpu_to_le16(val);
+ memcpy(net_buf_simple_push(buf, sizeof(val)), &val, sizeof(val));
+}
+
+void net_buf_simple_push_be16(struct net_buf_simple *buf, u16_t val)
+{
+ NET_BUF_SIMPLE_DBG("buf %p val %u", buf, val);
+
+ val = sys_cpu_to_be16(val);
+ memcpy(net_buf_simple_push(buf, sizeof(val)), &val, sizeof(val));
+}
+
+void net_buf_simple_push_u8(struct net_buf_simple *buf, u8_t val)
+{
+ u8_t *data = net_buf_simple_push(buf, 1);
+
+ *data = val;
+}
+
+void *net_buf_simple_pull(struct net_buf_simple *buf, size_t len)
+{
+ NET_BUF_SIMPLE_DBG("buf %p len %u", buf, len);
+
+ NET_BUF_SIMPLE_ASSERT(buf->len >= len);
+
+ buf->len -= len;
+ return buf->data += len;
+}
+
+u8_t net_buf_simple_pull_u8(struct net_buf_simple *buf)
+{
+ u8_t val;
+
+ val = buf->data[0];
+ net_buf_simple_pull(buf, 1);
+
+ return val;
+}
+
+u16_t net_buf_simple_pull_le16(struct net_buf_simple *buf)
+{
+ u16_t val;
+
+ val = UNALIGNED_GET((u16_t *)buf->data);
+ net_buf_simple_pull(buf, sizeof(val));
+
+ return sys_le16_to_cpu(val);
+}
+
+u16_t net_buf_simple_pull_be16(struct net_buf_simple *buf)
+{
+ u16_t val;
+
+ val = UNALIGNED_GET((u16_t *)buf->data);
+ net_buf_simple_pull(buf, sizeof(val));
+
+ return sys_be16_to_cpu(val);
+}
+
+u32_t net_buf_simple_pull_le32(struct net_buf_simple *buf)
+{
+ u32_t val;
+
+ val = UNALIGNED_GET((u32_t *)buf->data);
+ net_buf_simple_pull(buf, sizeof(val));
+
+ return sys_le32_to_cpu(val);
+}
+
+u32_t net_buf_simple_pull_be32(struct net_buf_simple *buf)
+{
+ u32_t val;
+
+ val = UNALIGNED_GET((u32_t *)buf->data);
+ net_buf_simple_pull(buf, sizeof(val));
+
+ return sys_be32_to_cpu(val);
+}
+
+size_t net_buf_simple_headroom(struct net_buf_simple *buf)
+{
+ return buf->data - buf->__buf;
+}
+
+size_t net_buf_simple_tailroom(struct net_buf_simple *buf)
+{
+ return buf->size - net_buf_simple_headroom(buf) - buf->len;
+}
+
+void net_buf_reset(struct net_buf *buf)
+{
+ NET_BUF_ASSERT(buf->flags == 0);
+ NET_BUF_ASSERT(buf->frags == NULL);
+
+ buf->len = 0;
+ buf->data = buf->__buf;
+}
+
+void net_buf_reserve(struct net_buf *buf, size_t reserve)
+{
+ NET_BUF_ASSERT(buf);
+ NET_BUF_ASSERT(buf->len == 0);
+ NET_BUF_DBG("buf %p reserve %u", buf, reserve);
+
+ buf->data = buf->__buf + reserve;
+}
+
+void net_buf_slist_put(sys_slist_t *list, struct net_buf *buf)
+{
+ struct net_buf *tail;
+ unsigned int key;
+
+ NET_BUF_ASSERT(list);
+ NET_BUF_ASSERT(buf);
+
+ for (tail = buf; tail->frags; tail = tail->frags) {
+ tail->flags |= NET_BUF_FRAGS;
+ }
+
+ key = irq_lock();
+ sys_slist_append_list(list, &buf->node, &tail->node);
+ irq_unlock(key);
+}
+
+struct net_buf *net_buf_slist_get(sys_slist_t *list)
+{
+ struct net_buf *buf, *frag;
+ unsigned int key;
+
+ NET_BUF_ASSERT(list);
+
+ key = irq_lock();
+ buf = (void *)sys_slist_get(list);
+ irq_unlock(key);
+
+ if (!buf) {
+ return NULL;
+ }
+
+ /* Get any fragments belonging to this buffer */
+ for (frag = buf; (frag->flags & NET_BUF_FRAGS); frag = frag->frags) {
+ key = irq_lock();
+ frag->frags = (void *)sys_slist_get(list);
+ irq_unlock(key);
+
+ NET_BUF_ASSERT(frag->frags);
+
+ /* The fragments flag is only for list-internal usage */
+ frag->flags &= ~NET_BUF_FRAGS;
+ }
+
+ /* Mark the end of the fragment list */
+ frag->frags = NULL;
+
+ return buf;
+}
+
+void net_buf_put(struct k_fifo *fifo, struct net_buf *buf)
+{
+ struct net_buf *tail;
+
+ //NET_BUF_ASSERT(fifo);
+ //NET_BUF_ASSERT(buf);
+
+ //ble_mesh_msg_t msg = {0};
+
+ //ble_mesh_task_post(ble_mesh_msg_t *msg, 0);
+ for (tail = buf; tail->frags; tail = tail->frags) {
+ tail->flags |= NET_BUF_FRAGS;
+ }
+
+ //k_fifo_put_list(fifo, buf, tail);
+}
+
+struct net_buf *net_buf_ref(struct net_buf *buf)
+{
+ NET_BUF_ASSERT(buf);
+
+ NET_BUF_DBG("buf %p (old) ref %u pool_id %p",
+ buf, buf->ref, buf->pool_id);
+ buf->ref++;
+ return buf;
+}
+
+struct net_buf *net_buf_clone(struct net_buf *buf, s32_t timeout)
+{
+ struct net_buf_pool *pool;
+ struct net_buf *clone;
+
+ NET_BUF_ASSERT(buf);
+
+ pool = buf->pool_id;//net_buf_pool_get(buf->pool_id);
+
+ clone = net_buf_alloc(pool, timeout);
+ if (!clone) {
+ return NULL;
+ }
+
+ net_buf_reserve(clone, net_buf_headroom(buf));
+
+ /* TODO: Add reference to the original buffer instead of copying it. */
+ memcpy(net_buf_add(clone, buf->len), buf->data, buf->len);
+
+ return clone;
+}
+
+struct net_buf *net_buf_frag_last(struct net_buf *buf)
+{
+ NET_BUF_ASSERT(buf);
+
+ while (buf->frags) {
+ buf = buf->frags;
+ }
+
+ return buf;
+}
+
+void net_buf_frag_insert(struct net_buf *parent, struct net_buf *frag)
+{
+ NET_BUF_ASSERT(parent);
+ NET_BUF_ASSERT(frag);
+
+ if (parent->frags) {
+ net_buf_frag_last(frag)->frags = parent->frags;
+ }
+ /* Take ownership of the fragment reference */
+ parent->frags = frag;
+}
+
+#if defined(CONFIG_NET_BUF_LOG)
+struct net_buf *net_buf_frag_del_debug(struct net_buf *parent,
+ struct net_buf *frag,
+ const char *func, int line)
+#else
+struct net_buf *net_buf_frag_del(struct net_buf *parent, struct net_buf *frag)
+#endif
+{
+ struct net_buf *next_frag;
+
+ NET_BUF_ASSERT(frag);
+
+ if (parent) {
+ NET_BUF_ASSERT(parent->frags);
+ NET_BUF_ASSERT(parent->frags == frag);
+ parent->frags = frag->frags;
+ }
+
+ next_frag = frag->frags;
+
+ frag->frags = NULL;
+
+#if defined(CONFIG_NET_BUF_LOG)
+ net_buf_unref_debug(frag, func, line);
+#else
+ net_buf_unref(frag);
+#endif
+
+ return next_frag;
+}
+
+#if defined(CONFIG_NET_BUF_LOG)
+void net_buf_unref_debug(struct net_buf *buf, const char *func, int line)
+#else
+void net_buf_unref(struct net_buf *buf)
+#endif
+{
+ NET_BUF_ASSERT(buf);
+
+ while (buf) {
+ struct net_buf *frags = buf->frags;
+ struct net_buf_pool *pool;
+
+#if defined(CONFIG_NET_BUF_LOG)
+ if (!buf->ref) {
+ NET_BUF_ERR("%s():%d: buf %p double free", func, line,
+ buf);
+ return;
+ }
+#endif
+ NET_BUF_DBG("buf %p ref %u pool_id %p frags %p", buf, buf->ref,
+ buf->pool_id, buf->frags);
+
+ /* Change by Espressif. Add !buf->ref to avoid minus 0 */
+ if (!buf->ref || --buf->ref > 0) {
+ return;
+ }
+
+ buf->frags = NULL;
+
+ pool = buf->pool_id;//net_buf_pool_get(buf->pool_id);
+ pool->uninit_count++;
+#if defined(CONFIG_NET_BUF_POOL_USAGE)
+ pool->avail_count++;
+ BT_DBG("%s, pool->avail_count = %d, pool->uninit_count = %d", __func__,
+ pool->avail_count, pool->uninit_count);
+ NET_BUF_ASSERT(pool->avail_count <= pool->buf_count);
+#endif
+
+ if (pool->destroy) {
+ pool->destroy(buf);
+ } else {
+ net_buf_destroy(buf);
+ }
+
+ buf = frags;
+ }
+}
+
+#if defined(CONFIG_NET_BUF_LOG)
+struct net_buf *net_buf_alloc_debug(struct net_buf_pool *pool, s32_t timeout,
+ const char *func, int line)
+#else
+struct net_buf *net_buf_alloc(struct net_buf_pool *pool, s32_t timeout)
+#endif
+{
+ struct net_buf *buf = NULL;
+ unsigned int key;
+ int i;
+
+ NET_BUF_ASSERT(pool);
+
+ NET_BUF_DBG("%s():%d: pool %p timeout %d", func, line, pool, timeout);
+ BT_DBG("%s, pool = %p , pool->uninit_count= %d, buf_count = %d", __func__,
+ pool, pool->uninit_count, pool->buf_count);
+ /* We need to lock interrupts temporarily to prevent race conditions
+ * when accessing pool->uninit_count.
+ */
+ key = irq_lock();
+
+ /* If there are uninitialized buffers we're guaranteed to succeed
+ * with the allocation one way or another.
+ */
+ if (pool->uninit_count) {
+ /* Change by Espressif. Use buf when buf->ref is 0 */
+ for (i = pool->buf_count; i > 0; i--) {
+ buf = pool_get_uninit(pool, i);
+ if (!buf->ref) {
+ goto success;
+ }
+ }
+ // u16_t uninit_count;
+
+ /* If this is not the first access to the pool, we can
+ * be opportunistic and try to fetch a previously used
+ * buffer from the LIFO with K_NO_WAIT.
+ */
+ //if (pool->uninit_count < pool->buf_count) {
+ // buf = &pool->__bufs[pool->buf_count - pool->uninit_count];
+ //if (buf) {
+ // irq_unlock(key);
+ // goto success;
+ //}
+ //}
+
+ // uninit_count = pool->uninit_count--;
+ // irq_unlock(key);
+
+ // buf = pool_get_uninit(pool, uninit_count);
+ // goto success;
+ }
+
+ irq_unlock(key);
+
+ if (!buf) {
+ BT_ERR("%s():Failed to get free buffer", __func__);
+ }
+ return buf;
+
+success:
+ NET_BUF_DBG("allocated buf %p", buf);
+ irq_unlock(key);
+
+ buf->ref = 1;
+ buf->index = pool->buf_count - i;
+ buf->flags = 0;
+ buf->frags = NULL;
+ net_buf_reset(buf);
+
+ pool->uninit_count--;
+#if defined(CONFIG_NET_BUF_POOL_USAGE)
+ pool->avail_count--;
+ NET_BUF_ASSERT(pool->avail_count >= 0);
+#endif
+
+ return buf;
+}
+
+int net_buf_id(struct net_buf *buf)
+{
+ struct net_buf_pool *pool = buf->pool_id;//net_buf_pool_get(buf->pool_id);
+ u8_t *pool_start = (u8_t *)pool->__bufs;
+ u8_t *buf_ptr = (u8_t *)buf;
+
+ return (buf_ptr - pool_start) / BUF_SIZE(pool);
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/mesh_core/mesh_hci.c b/components/bt/ble_mesh/mesh_core/mesh_hci.c
new file mode 100644
index 0000000000..3ba3bd3116
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_hci.c
@@ -0,0 +1,58 @@
+/*
+ * Copyright (c) 2017 Nordic Semiconductor ASA
+ * Copyright (c) 2015-2016 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "mesh_hci.h"
+#include "stack/bt_types.h"
+#include "device/controller.h"
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+
+struct bt_dev bt_dev;
+
+void mesh_hci_init(void)
+{
+ const uint8_t *features = controller_get_interface()->get_features_ble()->as_array;
+ if (features != NULL) {
+ // We just copy the ble feature here, we don't care the classic bt feature in the ble mesh system.
+ memcpy(bt_dev.features[0], features, 8);
+ }
+
+ /**TODO: We support 20ms non-connectable adv interval, but not hci 5.0, so we need to add another flag
+ * to indicate if we support 20ms adv interval
+ * */
+#ifdef CONFIG_HCI_5_0_VERSION
+ bt_dev.hci_version = BT_HCI_VERSION_5_0;
+#else
+ bt_dev.hci_version = controller_get_interface()->get_bt_version()->hci_version;
+#endif
+ bt_dev.lmp_version = controller_get_interface()->get_bt_version()->lmp_version;
+ bt_dev.hci_revision = controller_get_interface()->get_bt_version()->hci_revision;
+ bt_dev.lmp_subversion = controller_get_interface()->get_bt_version()->lmp_subversion;
+ bt_dev.manufacturer = controller_get_interface()->get_bt_version()->manufacturer;
+ // Still to copy the features to the ble features struct.
+ memcpy(bt_dev.le.features, features, 8);
+
+ const uint8_t *p = controller_get_interface()->get_ble_supported_states();
+ uint64_t states_fh = 0, states_sh = 0;
+ // We don't have the STREAM_TO_UINT64 macro, so we should used stream to u32 two times add then put them together.
+ STREAM_TO_UINT32(states_fh, p);
+ STREAM_TO_UINT32(states_sh, p);
+ bt_dev.le.states = (states_sh << 32) | states_fh;
+
+#if defined(CONFIG_BT_CONN)
+ bt_dev.le.mtu = controller_get_interface()->get_acl_data_size_ble();
+#endif /* CONFIG_BT_CONN */
+
+#if defined(CONFIG_BT_SMP)
+ bt_dev.le.rl_size = controller_get_interface()->get_ble_resolving_list_max_size();
+#endif /* #if defined(CONFIG_BT_SMP) */
+
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/mesh_kernel.c b/components/bt/ble_mesh/mesh_core/mesh_kernel.c
new file mode 100644
index 0000000000..c01a4d2615
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_kernel.c
@@ -0,0 +1,239 @@
+/*
+ * Copyright (c) 2016 Intel Corporation
+ * Copyright (c) 2016 Wind River Systems, Inc.
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+
+#include "mesh_kernel.h"
+#include "esp_timer.h"
+#include "osi/hash_map.h"
+#include "osi/alarm.h"
+#include "common/bt_trace.h"
+#include "common/bt_defs.h"
+#include "osi/hash_functions.h"
+#include "mesh_trace.h"
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+#include "provisioner_prov.h"
+
+static osi_mutex_t ble_mesh_alarm_lock;
+static osi_mutex_t mesh_irq_lock;
+static hash_map_t *ble_mesh_alarm_hash_map;
+static const size_t BLE_MESH_GENERAL_ALARM_HASH_MAP_SIZE = 20 + CONFIG_BT_MESH_PBA_SAME_TIME + \
+ CONFIG_BT_MESH_PBG_SAME_TIME;
+
+typedef struct alarm_t {
+ /* timer id point to here */
+ esp_timer_handle_t alarm_hdl;
+ osi_alarm_callback_t cb;
+ void *cb_data;
+ int64_t deadline_us;
+} osi_alarm_t;
+
+static void ble_mesh_alarm_cb(void *data)
+{
+ assert(data != NULL);
+ struct k_delayed_work *work = (struct k_delayed_work *)data;
+ work->work.handler(&work->work);
+ return;
+}
+
+unsigned int irq_lock(void)
+{
+#if defined(CONFIG_BT_MESH_IRQ_LOCK) && CONFIG_BT_MESH_IRQ_LOCK
+ unsigned int key = XTOS_SET_INTLEVEL(XCHAL_EXCM_LEVEL);
+ return key;
+#else /* !CONFIG_BT_MESH_IRQ_LOCK */
+ /* Change by Espressif. In BLE Mesh, in order to improve the real-time
+ * requirements of bt controller, we use task lock to replace IRQ lock.
+ */
+ osi_mutex_lock(&mesh_irq_lock, OSI_MUTEX_MAX_TIMEOUT);
+ return 0;
+#endif /*#if (CONFIG_BT_MESH_IRQ_LOCK) */
+}
+
+void irq_unlock(unsigned int key)
+{
+#if defined(CONFIG_BT_MESH_IRQ_LOCK) && CONFIG_BT_MESH_IRQ_LOCK
+ XTOS_RESTORE_INTLEVEL(key);
+#else /* !CONFIG_BT_MESH_IRQ_LOCK */
+ osi_mutex_unlock(&mesh_irq_lock);
+#endif /*#if (CONFIG_BT_MESH_IRQ_LOCK) && CONFIG_BT_MESH_IRQ_LOCK */
+}
+
+s64_t k_uptime_get(void)
+{
+ /** k_uptime_get_32 is in in milliseconds,
+ * but esp_timer_get_time is in microseconds
+ */
+ return (esp_timer_get_time() / 1000);
+}
+
+u32_t k_uptime_get_32(void)
+{
+ /** k_uptime_get_32 is in in milliseconds,
+ * but esp_timer_get_time is in microseconds
+ */
+ return (u32_t)(esp_timer_get_time() / 1000);
+}
+
+void k_sleep(s32_t duration)
+{
+ vTaskDelay(duration / portTICK_PERIOD_MS);
+ return;
+}
+
+void mesh_k_init(void)
+{
+ osi_mutex_new(&ble_mesh_alarm_lock);
+ osi_mutex_new(&mesh_irq_lock);
+ ble_mesh_alarm_hash_map = hash_map_new(BLE_MESH_GENERAL_ALARM_HASH_MAP_SIZE,
+ hash_function_pointer, NULL, (data_free_fn)osi_alarm_free, NULL);
+ assert(ble_mesh_alarm_hash_map != NULL);
+}
+
+void k_delayed_work_init(struct k_delayed_work *work, k_work_handler_t handler)
+{
+ assert(work != NULL && ble_mesh_alarm_hash_map != NULL);
+
+ k_work_init(&work->work, handler);
+ _init_timeout(&work->timeout, NULL);
+ work->work_q = NULL;
+
+ osi_alarm_t *alarm = NULL;
+
+ // Get the alarm for the timer list entry.
+ osi_mutex_lock(&ble_mesh_alarm_lock, OSI_MUTEX_MAX_TIMEOUT);
+ if (!hash_map_has_key(ble_mesh_alarm_hash_map, (void *)work)) {
+ alarm = osi_alarm_new("ble_mesh", ble_mesh_alarm_cb, (void *)work, 0);
+ if (alarm == NULL) {
+ LOG_ERROR("%s, Unable to create alarm", __func__);
+ return;
+ }
+ if (!hash_map_set(ble_mesh_alarm_hash_map, work, (void *)alarm)) {
+ LOG_ERROR("%s Unable to add the work timer to the mesh alarm hash map.", __func__);
+ }
+ }
+ osi_mutex_unlock(&ble_mesh_alarm_lock);
+
+ alarm = hash_map_get(ble_mesh_alarm_hash_map, work);
+ if (alarm == NULL) {
+ LOG_WARN("%s, Unable to find expected alarm in hash map", __func__);
+ return;
+ }
+
+ // Just init the work timer only, don't start it.
+ osi_alarm_cancel(alarm);
+ return;
+}
+
+int k_delayed_work_submit(struct k_delayed_work *work,
+ s32_t delay)
+{
+ assert(work != NULL);
+
+ osi_alarm_t *alarm = hash_map_get(ble_mesh_alarm_hash_map, (void *)work);
+ if (alarm == NULL) {
+ LOG_WARN("%s, Unable to find expected alarm in hash map", __func__);
+ return -EINVAL;
+ }
+
+ // Cancel the alarm first, before start the alarm.
+ osi_alarm_cancel(alarm);
+ osi_alarm_set(alarm, delay);
+ return 0;
+}
+
+int k_delayed_work_cancel(struct k_delayed_work *work)
+{
+ assert(work != NULL);
+
+ // Check if the work have been store in the ble_mesh timer list or not.
+ osi_alarm_t *alarm = hash_map_get(ble_mesh_alarm_hash_map, (void *)work);
+ if (alarm == NULL) {
+ LOG_WARN("%s, Unable to find expected alarm in hash map", __func__);
+ return -EINVAL;
+ }
+
+ osi_alarm_cancel(alarm);
+ alarm->deadline_us = 0;
+ return 0;
+}
+
+int k_delayed_work_free(struct k_delayed_work *work)
+{
+ assert(work != NULL);
+
+ // Get the alarm for the timer list entry.
+ osi_alarm_t *alarm = hash_map_get(ble_mesh_alarm_hash_map, work);
+ if (alarm == NULL) {
+ LOG_WARN("%s Unable to find expected alarm in hash map", __func__);
+ return -EINVAL;
+ }
+
+ hash_map_erase(ble_mesh_alarm_hash_map, work);
+ return 0;
+}
+
+s32_t k_delayed_work_remaining_get(struct k_delayed_work *work)
+{
+ assert(work != NULL);
+
+ osi_alarm_t *alarm = hash_map_get(ble_mesh_alarm_hash_map, (void *)work);
+ if (alarm == NULL) {
+ LOG_WARN("%s Unable to find expected alarm in hash map", __func__);
+ return 0;
+ }
+
+ if (!alarm->deadline_us) {
+ return 0;
+ }
+
+ s32_t remain_time = 0;
+ int64_t now = esp_timer_get_time();
+ if ((alarm->deadline_us - now) < 0x1FFFFFFFFFF) {
+ remain_time = (alarm->deadline_us - now) / 1000;
+ } else {
+ return 0;
+ }
+
+ return remain_time;
+}
+
+void k_sem_give(struct k_sem *sem)
+{
+ assert(sem != NULL);
+ osi_mutex_unlock(sem->mutex);
+ return;
+}
+
+void k_sem_init(struct k_sem *sem, unsigned int initial_count,
+ unsigned int limit)
+{
+ assert(sem != NULL);
+
+ UNUSED(initial_count);
+ UNUSED(limit);
+
+ sem->mutex = xSemaphoreCreateBinary();
+ if (sem->mutex == NULL) {
+ LOG_WARN("%s, the mutex alloc fail", __func__);
+ return;
+ }
+
+ return;
+}
+
+int k_sem_take(struct k_sem *sem, s32_t timeout)
+{
+ assert(sem != NULL);
+ return osi_mutex_lock(sem->mutex, timeout);
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/mesh_main.c b/components/bt/ble_mesh/mesh_core/mesh_main.c
new file mode 100644
index 0000000000..0684dc5ecf
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_main.c
@@ -0,0 +1,449 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG)
+#include "mesh_trace.h"
+
+#include "mesh_main.h"
+#include "adv.h"
+#include "prov.h"
+#include "net.h"
+#include "beacon.h"
+#include "lpn.h"
+#include "friend.h"
+#include "transport.h"
+#include "access.h"
+#include "foundation.h"
+#include "proxy.h"
+#include "mesh.h"
+#include "mesh_hci.h"
+#include "settings.h"
+#include "provisioner_prov.h"
+#include "provisioner_proxy.h"
+#include "provisioner_main.h"
+
+static volatile bool provisioner_en;
+
+#define ACTION_ENTER 0x01
+#define ACTION_SUSPEND 0x02
+#define ACTION_EXIT 0x03
+
+#if CONFIG_BT_MESH_NODE
+
+int bt_mesh_provision(const u8_t net_key[16], u16_t net_idx,
+ u8_t flags, u32_t iv_index, u32_t seq,
+ u16_t addr, const u8_t dev_key[16])
+{
+ int err;
+
+ BT_DBG("Primary Element: 0x%04x", addr);
+ BT_DBG("net_idx 0x%04x flags 0x%02x iv_index 0x%04x",
+ net_idx, flags, iv_index);
+ BT_DBG("Device key: %s", bt_hex(dev_key, 16));
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT)) {
+ bt_mesh_proxy_prov_disable();
+ }
+
+ err = bt_mesh_net_create(net_idx, flags, net_key, iv_index);
+ if (err) {
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT)) {
+ bt_mesh_proxy_prov_enable();
+ }
+
+ return err;
+ }
+
+ bt_mesh.seq = seq;
+
+ bt_mesh_comp_provision(addr);
+
+ memcpy(bt_mesh.dev_key, dev_key, 16);
+
+ if (bt_mesh_beacon_get() == BT_MESH_BEACON_ENABLED) {
+ bt_mesh_beacon_enable();
+ } else {
+ bt_mesh_beacon_disable();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ BT_DBG("Storing network information persistently");
+ bt_mesh_store_net();
+ bt_mesh_store_subnet(&bt_mesh.sub[0]);
+ bt_mesh_store_iv();
+ }
+
+ BT_DBG("%s, gatt_proxy = %d", __func__, bt_mesh_gatt_proxy_get());
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ bt_mesh_gatt_proxy_get() != BT_MESH_GATT_PROXY_NOT_SUPPORTED) {
+ bt_mesh_proxy_gatt_enable();
+ bt_mesh_adv_update();
+ }
+
+ /* Add this to avoid "already active status" for bt_mesh_scan_enable() */
+ bt_mesh_scan_disable();
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_init();
+ } else {
+ bt_mesh_scan_enable();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ bt_mesh_friend_init();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PROV)) {
+ bt_mesh_prov_complete(net_idx, addr, flags, iv_index);
+ }
+
+ return 0;
+}
+
+void bt_mesh_reset(void)
+{
+ if (!bt_mesh.valid) {
+ BT_WARN("%s: not provisioned", __func__);
+ return;
+ }
+
+ bt_mesh_comp_unprovision();
+
+ bt_mesh.iv_index = 0;
+ bt_mesh.seq = 0;
+ bt_mesh.iv_update = 0;
+ bt_mesh.pending_update = 0;
+ bt_mesh.valid = 0;
+ bt_mesh.last_update = 0;
+ bt_mesh.ivu_initiator = 0;
+
+ k_delayed_work_cancel(&bt_mesh.ivu_complete);
+
+ bt_mesh_cfg_reset();
+
+ bt_mesh_rx_reset();
+ bt_mesh_tx_reset();
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_disable(true);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ bt_mesh_friend_clear_net_idx(BT_MESH_KEY_ANY);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ bt_mesh_proxy_gatt_disable();
+ }
+
+#if defined(CONFIG_BT_MESH_SETTINGS)
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_clear_net();
+ }
+#endif
+ memset(bt_mesh.dev_key, 0, sizeof(bt_mesh.dev_key));
+
+ memset(bt_mesh.rpl, 0, sizeof(bt_mesh.rpl));
+
+ bt_mesh_scan_disable();
+ bt_mesh_beacon_disable();
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PROV)) {
+ bt_mesh_prov_reset();
+ }
+}
+
+bool bt_mesh_is_provisioned(void)
+{
+ return bt_mesh.valid; // Previous is "bool provisioned"
+}
+
+int bt_mesh_prov_enable(bt_mesh_prov_bearer_t bearers)
+{
+ if (bt_mesh_is_provisioned()) {
+ return -EALREADY;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_ADV) &&
+ (bearers & BT_MESH_PROV_ADV)) {
+ /* Make sure the scanning is for provisioning invitations */
+ bt_mesh_scan_enable();
+ /* Enable unprovisioned beacon sending */
+ bt_mesh_beacon_enable();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ (bearers & BT_MESH_PROV_GATT)) {
+ bt_mesh_proxy_prov_enable();
+ bt_mesh_adv_update();
+ }
+
+ provisioner_en = false;
+
+ return 0;
+}
+
+int bt_mesh_prov_disable(bt_mesh_prov_bearer_t bearers)
+{
+ if (bt_mesh_is_provisioned()) {
+ return -EALREADY;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_ADV) &&
+ (bearers & BT_MESH_PROV_ADV)) {
+ bt_mesh_beacon_disable();
+ bt_mesh_scan_disable();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ (bearers & BT_MESH_PROV_GATT)) {
+ bt_mesh_proxy_prov_disable();
+ bt_mesh_adv_update();
+ }
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_NODE */
+
+bool bt_mesh_is_provisioner_en(void)
+{
+ return provisioner_en;
+}
+
+#if CONFIG_BT_MESH_PROVISIONER
+
+int bt_mesh_provisioner_enable(bt_mesh_prov_bearer_t bearers)
+{
+ int err;
+
+ if (bt_mesh_is_provisioner_en()) {
+ BT_ERR("Provisioner already enabled");
+ return -EALREADY;
+ }
+
+ err = provisioner_upper_init();
+ if (err) {
+ BT_ERR("%s: provisioner_upper_init fail", __func__);
+ return err;
+ }
+
+ if ((IS_ENABLED(CONFIG_BT_MESH_PB_ADV) &&
+ (bearers & BT_MESH_PROV_ADV)) ||
+ (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ (bearers & BT_MESH_PROV_GATT))) {
+ bt_mesh_scan_enable();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ (bearers & BT_MESH_PROV_GATT)) {
+ provisioner_pb_gatt_enable();
+ }
+
+ provisioner_en = true;
+
+ return 0;
+}
+
+int bt_mesh_provisioner_disable(bt_mesh_prov_bearer_t bearers)
+{
+ if (!bt_mesh_is_provisioner_en()) {
+ BT_ERR("Provisioner already disabled");
+ return -EALREADY;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ (bearers & BT_MESH_PROV_GATT)) {
+ provisioner_pb_gatt_disable();
+ }
+
+ if ((IS_ENABLED(CONFIG_BT_MESH_PB_ADV) &&
+ (bearers & BT_MESH_PROV_ADV)) &&
+ (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ (bearers & BT_MESH_PROV_GATT))) {
+ bt_mesh_scan_disable();
+ }
+
+ provisioner_en = false;
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+/* The following API is for fast provisioning */
+
+#if CONFIG_BT_MESH_FAST_PROV
+
+u8_t bt_mesh_set_fast_prov_action(u8_t action)
+{
+ if (!action || action > ACTION_EXIT) {
+ return 0x01;
+ }
+
+ if ((!provisioner_en && (action == ACTION_SUSPEND || action == ACTION_EXIT)) ||
+ (provisioner_en && (action == ACTION_ENTER))) {
+ BT_WARN("%s: action is already done", __func__);
+ return 0x0;
+ }
+
+ if (action == ACTION_ENTER) {
+#if 0
+ /* If the device is provisioned using PB-GATT and connected to
+ * the phone with proxy service, proxy_gatt shall not be disabled
+ * here. The node needs to send some status messages to the phone
+ * while it is connected.
+ */
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ bt_mesh_proxy_gatt_disable();
+ }
+#endif
+ if (bt_mesh_beacon_get() == BT_MESH_BEACON_ENABLED) {
+ bt_mesh_beacon_disable();
+ }
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT)) {
+ provisioner_pb_gatt_enable();
+ }
+ provisioner_set_fast_prov_flag(true);
+ provisioner_en = true;
+ } else {
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT)) {
+ provisioner_pb_gatt_disable();
+ }
+ if (bt_mesh_beacon_get() == BT_MESH_BEACON_ENABLED) {
+ bt_mesh_beacon_enable();
+ }
+#if 0
+ /* Mesh Proxy GATT will be re-enabled on application layer */
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ bt_mesh_gatt_proxy_get() != BT_MESH_GATT_PROXY_NOT_SUPPORTED) {
+ bt_mesh_proxy_gatt_enable();
+ bt_mesh_adv_update();
+ }
+#endif
+ provisioner_set_fast_prov_flag(false);
+ provisioner_en = false;
+ if (action == ACTION_EXIT) {
+ provisioner_upper_reset_all_nodes();
+ provisioner_prov_reset_all_nodes();
+ }
+ }
+
+ return 0x0;
+}
+
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+int bt_mesh_init(const struct bt_mesh_prov *prov,
+ const struct bt_mesh_comp *comp)
+{
+ int err;
+
+ //err = bt_mesh_test();
+ //if (err) {
+ // return err;
+ //}
+
+ // The mesh kernel should be initialized first.
+ mesh_k_init();
+
+ mesh_hci_init();
+
+ bt_mesh_adapt_init();
+
+ err = bt_mesh_comp_register(comp);
+ if (err) {
+ return err;
+ }
+
+ /** GATT is initialised here in order to register services later */
+ bt_mesh_gatt_init();
+
+ /** Register service in the init function,
+ * then start it according to bt_mesh_proxy_gatt_enable
+ */
+#if CONFIG_BT_MESH_NODE
+ extern struct bt_gatt_service proxy_svc;
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ bt_mesh_gatts_service_register(&proxy_svc);
+ }
+
+ /** Register service in the init function,
+ * then start it according to bt_mesh_proxy_prov_enable
+ */
+ extern struct bt_gatt_service prov_svc;
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT)) {
+ bt_mesh_gatts_service_register(&prov_svc);
+ }
+#endif
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PROV)) {
+#if CONFIG_BT_MESH_NODE
+ err = bt_mesh_prov_init(prov);
+ if (err) {
+ return err;
+ }
+#endif
+#if CONFIG_BT_MESH_PROVISIONER
+ err = provisioner_prov_init(prov);
+ if (err) {
+ return err;
+ }
+#endif
+ }
+
+ bt_mesh_net_init();
+ bt_mesh_trans_init();
+#if CONFIG_BT_MESH_NODE
+ /* Changed by Espressif, add random delay (0 ~ 3s) */
+#if defined(CONFIG_BT_MESH_FAST_PROV)
+ u32_t delay = 0;
+ bt_mesh_rand(&delay, sizeof(u32_t));
+ vTaskDelay((delay % 3000) / portTICK_PERIOD_MS);
+#endif
+ bt_mesh_beacon_init();
+#endif
+ bt_mesh_adv_init();
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PROXY)) {
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_proxy_init();
+#endif
+#if CONFIG_BT_MESH_PROVISIONER
+ provisioner_proxy_init();
+#endif
+ }
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ /* If node & provisioner are both enabled and the
+ * device starts as a node, it must finish provisioning */
+ err = provisioner_upper_init();
+ if (err) {
+ return err;
+ }
+#endif
+
+#if defined(CONFIG_BT_MESH_SETTINGS)
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_settings_init();
+ }
+#endif
+
+ return 0;
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/mesh_slist.c b/components/bt/ble_mesh/mesh_core/mesh_slist.c
new file mode 100644
index 0000000000..d80fddfc8b
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_slist.c
@@ -0,0 +1,317 @@
+/*
+ * Copyright (c) 2016 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "mesh_slist.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+
+static sys_slist_t test_list;
+
+struct container_node {
+ sys_snode_t node;
+ int unused;
+};
+
+static struct container_node test_node_1;
+static struct container_node test_node_2;
+static struct container_node test_node_3;
+static struct container_node test_node_4;
+
+static inline bool verify_emptyness(sys_slist_t *list)
+{
+ sys_snode_t *node;
+ sys_snode_t *s_node;
+ struct container_node *cnode;
+ struct container_node *s_cnode;
+ int count;
+
+ if (!sys_slist_is_empty(list)) {
+ return false;
+ }
+
+ if (sys_slist_peek_head(list)) {
+ return false;
+ }
+
+ if (sys_slist_peek_tail(list)) {
+ return false;
+ }
+
+ count = 0;
+ SYS_SLIST_FOR_EACH_NODE(list, node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ SYS_SLIST_FOR_EACH_NODE_SAFE(list, node, s_node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ count = 0;
+ SYS_SLIST_FOR_EACH_CONTAINER(list, cnode, node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ count = 0;
+ SYS_SLIST_FOR_EACH_CONTAINER_SAFE(list, cnode, s_cnode, node) {
+ count++;
+ }
+
+ if (count) {
+ return false;
+ }
+
+ return true;
+}
+
+static inline bool verify_content_amount(sys_slist_t *list, int amount)
+{
+ sys_snode_t *node;
+ sys_snode_t *s_node;
+ struct container_node *cnode;
+ struct container_node *s_cnode;
+ int count;
+
+ if (sys_slist_is_empty(list)) {
+ return false;
+ }
+
+ if (!sys_slist_peek_head(list)) {
+ return false;
+ }
+
+ if (!sys_slist_peek_tail(list)) {
+ return false;
+ }
+
+ count = 0;
+ SYS_SLIST_FOR_EACH_NODE(list, node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ count = 0;
+ SYS_SLIST_FOR_EACH_NODE_SAFE(list, node, s_node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ count = 0;
+ SYS_SLIST_FOR_EACH_CONTAINER(list, cnode, node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ count = 0;
+ SYS_SLIST_FOR_EACH_CONTAINER_SAFE(list, cnode, s_cnode, node) {
+ count++;
+ }
+
+ if (count != amount) {
+ return false;
+ }
+
+ return true;
+}
+
+static inline bool verify_tail_head(sys_slist_t *list,
+ sys_snode_t *head,
+ sys_snode_t *tail,
+ bool same)
+{
+ if (sys_slist_peek_head(list) != head) {
+ return false;
+ }
+
+ if (sys_slist_peek_tail(list) != tail) {
+ return false;
+ }
+
+ if (same) {
+ if (sys_slist_peek_head(list) != sys_slist_peek_tail(list)) {
+ return false;
+ }
+ } else {
+ if (sys_slist_peek_head(list) == sys_slist_peek_tail(list)) {
+ return false;
+ }
+ }
+
+ return true;
+}
+
+void slist_test(void)
+{
+ sys_slist_init(&test_list);
+
+ //zassert_true((verify_emptyness(&test_list)), "test_list should be empty");
+
+ /* Appending node 1 */
+ sys_slist_append(&test_list, &test_node_1.node);
+ //zassert_true((verify_content_amount(&test_list, 1)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_1.node,
+ // &test_node_1.node, true)),
+ // "test_list head/tail are wrong");
+
+ /* Finding and removing node 1 */
+ sys_slist_find_and_remove(&test_list, &test_node_1.node);
+ //zassert_true((verify_emptyness(&test_list)),
+ // "test_list should be empty");
+
+ /* Prepending node 1 */
+ sys_slist_prepend(&test_list, &test_node_1.node);
+ //zassert_true((verify_content_amount(&test_list, 1)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_1.node,
+ // &test_node_1.node, true)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 1 */
+ sys_slist_remove(&test_list, NULL, &test_node_1.node);
+ //zassert_true((verify_emptyness(&test_list)),
+ // "test_list should be empty");
+
+ /* Appending node 1 */
+ sys_slist_append(&test_list, &test_node_1.node);
+ /* Prepending node 2 */
+ sys_slist_prepend(&test_list, &test_node_2.node);
+
+ //zassert_true((verify_content_amount(&test_list, 2)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_1.node, false)),
+ // "test_list head/tail are wrong");
+
+ /* Appending node 3 */
+ sys_slist_append(&test_list, &test_node_3.node);
+
+ //zassert_true((verify_content_amount(&test_list, 3)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_3.node, false)),
+ // "test_list head/tail are wrong");
+
+ //zassert_true((sys_slist_peek_next(&test_node_2.node) ==
+ // &test_node_1.node),
+ // "test_list node links are wrong");
+
+ /* Inserting node 4 after node 2 */
+ sys_slist_insert(&test_list, &test_node_2.node, &test_node_4.node);
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_3.node, false)),
+ // "test_list head/tail are wrong");
+
+ //zassert_true((sys_slist_peek_next(&test_node_2.node) ==
+ // &test_node_4.node),
+ // "test_list node links are wrong");
+
+ /* Finding and removing node 1 */
+ sys_slist_find_and_remove(&test_list, &test_node_1.node);
+ //zassert_true((verify_content_amount(&test_list, 3)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_3.node, false)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 3 */
+ sys_slist_remove(&test_list, &test_node_4.node, &test_node_3.node);
+ //zassert_true((verify_content_amount(&test_list, 2)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_4.node, false)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 4 */
+ sys_slist_remove(&test_list, &test_node_2.node, &test_node_4.node);
+ //zassert_true((verify_content_amount(&test_list, 1)),
+ // "test_list has wrong content");
+
+ //zassert_true((verify_tail_head(&test_list, &test_node_2.node,
+ // &test_node_2.node, true)),
+ // "test_list head/tail are wrong");
+
+ /* Removing node 2 */
+ sys_slist_remove(&test_list, NULL, &test_node_2.node);
+ //zassert_true((verify_emptyness(&test_list)),
+ // "test_list should be empty");
+
+ /* test iterator from a node */
+ struct data_node {
+ sys_snode_t node;
+ int data;
+ } data_node[6] = {
+ { .data = 0 },
+ { .data = 1 },
+ { .data = 2 },
+ { .data = 3 },
+ { .data = 4 },
+ { .data = 5 },
+ };
+ sys_snode_t *node = NULL;
+ int ii;
+
+ sys_slist_init(&test_list);
+
+ for (ii = 0; ii < 6; ii++) {
+ sys_slist_append(&test_list, &data_node[ii].node);
+ }
+
+ ii = 0;
+ SYS_SLIST_ITERATE_FROM_NODE(&test_list, node) {
+ ii++;
+ if (((struct data_node *)node)->data == 2) {
+ break;
+ }
+ }
+ //zassert_equal(ii, 3, "");
+
+ ii = 0;
+ SYS_SLIST_ITERATE_FROM_NODE(&test_list, node) {
+ ii++;
+ if (((struct data_node *)node)->data == 3) {
+ break;
+ }
+ }
+ //zassert_equal(ii, 1, "");
+
+ ii = 0;
+ SYS_SLIST_ITERATE_FROM_NODE(&test_list, node) {
+ ii++;
+ }
+ //zassert_equal(ii, 2, "");
+}
+
+#endif /* #if CONFIG_BT_MESH */
+
diff --git a/components/bt/ble_mesh/mesh_core/mesh_util.c b/components/bt/ble_mesh/mesh_core/mesh_util.c
new file mode 100644
index 0000000000..e95676575b
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/mesh_util.c
@@ -0,0 +1,89 @@
+/*
+ * Copyright (c) 2017 Nordic Semiconductor ASA
+ * Copyright (c) 2016 Vinayak Kariappa Chettimada
+ * Copyright (c) 2015-2016 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+#include "mesh_util.h"
+#include "mesh_kernel.h"
+#include "mesh_aes_encrypt.h"
+
+#define MASK_TWENTY_SEVEN 0x1b
+
+const char *bt_hex(const void *buf, size_t len)
+{
+ static const char hex[] = "0123456789abcdef";
+ static char hexbufs[4][129];
+ static u8_t curbuf;
+ const u8_t *b = buf;
+ unsigned int mask;
+ char *str;
+ int i;
+
+ mask = irq_lock();
+ str = hexbufs[curbuf++];
+ curbuf %= ARRAY_SIZE(hexbufs);
+ irq_unlock(mask);
+
+ len = min(len, (sizeof(hexbufs[0]) - 1) / 2);
+
+ for (i = 0; i < len; i++) {
+ str[i * 2] = hex[b[i] >> 4];
+ str[i * 2 + 1] = hex[b[i] & 0xf];
+ }
+
+ str[i * 2] = '\0';
+
+ return str;
+}
+
+void mem_rcopy(u8_t *dst, u8_t const *src, u16_t len)
+{
+ src += len;
+ while (len--) {
+ *dst++ = *--src;
+ }
+}
+
+unsigned int _copy(uint8_t *to, unsigned int to_len,
+ const uint8_t *from, unsigned int from_len)
+{
+ if (from_len <= to_len) {
+ (void)memcpy(to, from, from_len);
+ return from_len;
+ } else {
+ return TC_CRYPTO_FAIL;
+ }
+}
+
+void _set(void *to, uint8_t val, unsigned int len)
+{
+ (void)memset(to, val, len);
+}
+
+/*
+ * Doubles the value of a byte for values up to 127.
+ */
+uint8_t _double_byte(uint8_t a)
+{
+ return ((a << 1) ^ ((a >> 7) * MASK_TWENTY_SEVEN));
+}
+
+int _compare(const uint8_t *a, const uint8_t *b, size_t size)
+{
+ const uint8_t *tempa = a;
+ const uint8_t *tempb = b;
+ uint8_t result = 0;
+
+ for (unsigned int i = 0; i < size; i++) {
+ result |= tempa[i] ^ tempb[i];
+ }
+ return result;
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/net.c b/components/bt/ble_mesh/mesh_core/net.c
new file mode 100644
index 0000000000..be7f753954
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/net.c
@@ -0,0 +1,1509 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+#include "mesh_util.h"
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+
+#include "mesh_main.h"
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_NET)
+#include "mesh_trace.h"
+
+#include "crypto.h"
+#include "adv.h"
+#include "mesh.h"
+#include "net.h"
+#include "lpn.h"
+#include "friend.h"
+#include "proxy.h"
+#include "transport.h"
+#include "access.h"
+#include "foundation.h"
+#include "beacon.h"
+#include "settings.h"
+#include "prov.h"
+
+#include "common.h"
+
+/* Minimum valid Mesh Network PDU length. The Network headers
+ * themselves take up 9 bytes. After that there is a minumum of 1 byte
+ * payload for both CTL=1 and CTL=0 PDUs (smallest OpCode is 1 byte). CTL=1
+ * PDUs must use a 64-bit (8 byte) NetMIC, whereas CTL=0 PDUs have at least
+ * a 32-bit (4 byte) NetMIC and AppMIC giving again a total of 8 bytes.
+ */
+#define BT_MESH_NET_MIN_PDU_LEN (BT_MESH_NET_HDR_LEN + 1 + 8)
+
+/* Seq limit after IV Update is triggered */
+#define IV_UPDATE_SEQ_LIMIT 8000000
+
+#if defined(CONFIG_BT_MESH_IV_UPDATE_TEST)
+/* Small test timeout for IV Update Procedure testing */
+#define IV_UPDATE_TIMEOUT K_SECONDS(120)
+#else
+/* Maximum time to stay in IV Update mode (96 < time < 144) */
+#define IV_UPDATE_TIMEOUT K_HOURS(120)
+#endif /* CONFIG_BT_MESH_IV_UPDATE_TEST */
+
+#define IVI(pdu) ((pdu)[0] >> 7)
+#define NID(pdu) ((pdu)[0] & 0x7f)
+#define CTL(pdu) ((pdu)[1] >> 7)
+#define TTL(pdu) ((pdu)[1] & 0x7f)
+#define SEQ(pdu) (((u32_t)(pdu)[2] << 16) | \
+ ((u32_t)(pdu)[3] << 8) | (u32_t)(pdu)[4]);
+#define SRC(pdu) (sys_get_be16(&(pdu)[5]))
+#define DST(pdu) (sys_get_be16(&(pdu)[7]))
+
+/* Determine how many friendship credentials are needed */
+#if defined(CONFIG_BT_MESH_FRIEND)
+#define FRIEND_CRED_COUNT CONFIG_BT_MESH_FRIEND_LPN_COUNT
+#elif defined(CONFIG_BT_MESH_LOW_POWER)
+#define FRIEND_CRED_COUNT CONFIG_BT_MESH_SUBNET_COUNT
+#else
+#define FRIEND_CRED_COUNT 0
+#endif
+
+#if FRIEND_CRED_COUNT > 0
+static struct friend_cred friend_cred[FRIEND_CRED_COUNT];
+#endif
+
+static u64_t msg_cache[CONFIG_BT_MESH_MSG_CACHE_SIZE];
+static u16_t msg_cache_next;
+
+/* Singleton network context (the implementation only supports one) */
+struct bt_mesh_net bt_mesh = {
+ .local_queue = SYS_SLIST_STATIC_INIT(&bt_mesh.local_queue),
+ .sub = {
+ [0 ... (CONFIG_BT_MESH_SUBNET_COUNT - 1)] = {
+ .net_idx = BT_MESH_KEY_UNUSED,
+ }
+ },
+ .app_keys = {
+ [0 ... (CONFIG_BT_MESH_APP_KEY_COUNT - 1)] = {
+ .net_idx = BT_MESH_KEY_UNUSED,
+ }
+ },
+};
+
+static u32_t dup_cache[4];
+static int dup_cache_next;
+
+static bool check_dup(struct net_buf_simple *data)
+{
+ const u8_t *tail = net_buf_simple_tail(data);
+ u32_t val;
+ int i;
+
+ val = sys_get_be32(tail - 4) ^ sys_get_be32(tail - 8);
+
+ for (i = 0; i < ARRAY_SIZE(dup_cache); i++) {
+ if (dup_cache[i] == val) {
+ return true;
+ }
+ }
+
+ dup_cache[dup_cache_next++] = val;
+ dup_cache_next %= ARRAY_SIZE(dup_cache);
+
+ return false;
+}
+
+static u64_t msg_hash(struct bt_mesh_net_rx *rx, struct net_buf_simple *pdu)
+{
+ u32_t hash1, hash2;
+
+ /* Three least significant bytes of IVI + first byte of SEQ */
+ hash1 = (BT_MESH_NET_IVI_RX(rx) << 8) | pdu->data[2];
+
+ /* Two last bytes of SEQ + SRC */
+ memcpy(&hash2, &pdu->data[3], 4);
+
+ return (u64_t)hash1 << 32 | (u64_t)hash2;
+}
+
+static bool msg_cache_match(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *pdu)
+{
+ u64_t hash = msg_hash(rx, pdu);
+ u16_t i;
+
+ for (i = 0; i < ARRAY_SIZE(msg_cache); i++) {
+ if (msg_cache[i] == hash) {
+ return true;
+ }
+ }
+
+ /* Add to the cache */
+ msg_cache[msg_cache_next++] = hash;
+ msg_cache_next %= ARRAY_SIZE(msg_cache);
+
+ return false;
+}
+
+struct bt_mesh_subnet *bt_mesh_subnet_get(u16_t net_idx)
+{
+ int i;
+
+ if (net_idx == BT_MESH_KEY_ANY) {
+ return &bt_mesh.sub[0];
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ if (bt_mesh.sub[i].net_idx == net_idx) {
+ return &bt_mesh.sub[i];
+ }
+ }
+
+ return NULL;
+}
+
+int bt_mesh_net_keys_create(struct bt_mesh_subnet_keys *keys,
+ const u8_t key[16])
+{
+ u8_t p[] = { 0 };
+ u8_t nid;
+ int err;
+
+ err = bt_mesh_k2(key, p, sizeof(p), &nid, keys->enc, keys->privacy);
+ if (err) {
+ BT_ERR("Unable to generate NID, EncKey & PrivacyKey");
+ return err;
+ }
+
+ memcpy(keys->net, key, 16);
+
+ keys->nid = nid;
+
+ BT_DBG("NID 0x%02x EncKey %s", keys->nid, bt_hex(keys->enc, 16));
+ BT_DBG("PrivacyKey %s", bt_hex(keys->privacy, 16));
+
+ err = bt_mesh_k3(key, keys->net_id);
+ if (err) {
+ BT_ERR("Unable to generate Net ID");
+ return err;
+ }
+
+ BT_DBG("NetID %s", bt_hex(keys->net_id, 8));
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ err = bt_mesh_identity_key(key, keys->identity);
+ if (err) {
+ BT_ERR("Unable to generate IdentityKey");
+ return err;
+ }
+
+ BT_DBG("IdentityKey %s", bt_hex(keys->identity, 16));
+#endif /* GATT_PROXY */
+
+ err = bt_mesh_beacon_key(key, keys->beacon);
+ if (err) {
+ BT_ERR("Unable to generate beacon key");
+ return err;
+ }
+
+ BT_DBG("BeaconKey %s", bt_hex(keys->beacon, 16));
+
+ return 0;
+}
+
+#if (defined(CONFIG_BT_MESH_LOW_POWER) || \
+ defined(CONFIG_BT_MESH_FRIEND))
+int friend_cred_set(struct friend_cred *cred, u8_t idx, const u8_t net_key[16])
+{
+ u16_t lpn_addr, frnd_addr;
+ int err;
+ u8_t p[9];
+
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+ if (cred->addr == bt_mesh.lpn.frnd) {
+ lpn_addr = bt_mesh_primary_addr();
+ frnd_addr = cred->addr;
+ } else {
+ lpn_addr = cred->addr;
+ frnd_addr = bt_mesh_primary_addr();
+ }
+#else
+ lpn_addr = cred->addr;
+ frnd_addr = bt_mesh_primary_addr();
+#endif
+
+ BT_DBG("LPNAddress 0x%04x FriendAddress 0x%04x", lpn_addr, frnd_addr);
+ BT_DBG("LPNCounter 0x%04x FriendCounter 0x%04x", cred->lpn_counter,
+ cred->frnd_counter);
+
+ p[0] = 0x01;
+ sys_put_be16(lpn_addr, p + 1);
+ sys_put_be16(frnd_addr, p + 3);
+ sys_put_be16(cred->lpn_counter, p + 5);
+ sys_put_be16(cred->frnd_counter, p + 7);
+
+ err = bt_mesh_k2(net_key, p, sizeof(p), &cred->cred[idx].nid,
+ cred->cred[idx].enc, cred->cred[idx].privacy);
+ if (err) {
+ BT_ERR("Unable to generate NID, EncKey & PrivacyKey");
+ return err;
+ }
+
+ BT_DBG("Friend NID 0x%02x EncKey %s", cred->cred[idx].nid,
+ bt_hex(cred->cred[idx].enc, 16));
+ BT_DBG("Friend PrivacyKey %s", bt_hex(cred->cred[idx].privacy, 16));
+
+ return 0;
+}
+
+void friend_cred_refresh(u16_t net_idx)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(friend_cred); i++) {
+ struct friend_cred *cred = &friend_cred[i];
+
+ if (cred->addr != BT_MESH_ADDR_UNASSIGNED &&
+ cred->net_idx == net_idx) {
+ memcpy(&cred->cred[0], &cred->cred[1],
+ sizeof(cred->cred[0]));
+ }
+ }
+}
+
+int friend_cred_update(struct bt_mesh_subnet *sub)
+{
+ int err, i;
+
+ BT_DBG("net_idx 0x%04x", sub->net_idx);
+
+ for (i = 0; i < ARRAY_SIZE(friend_cred); i++) {
+ struct friend_cred *cred = &friend_cred[i];
+
+ if (cred->addr == BT_MESH_ADDR_UNASSIGNED ||
+ cred->net_idx != sub->net_idx) {
+ continue;
+ }
+
+ err = friend_cred_set(cred, 1, sub->keys[1].net);
+ if (err) {
+ return err;
+ }
+ }
+
+ return 0;
+}
+
+struct friend_cred *friend_cred_create(struct bt_mesh_subnet *sub, u16_t addr,
+ u16_t lpn_counter, u16_t frnd_counter)
+{
+ struct friend_cred *cred;
+ int i, err;
+
+ BT_DBG("net_idx 0x%04x addr 0x%04x", sub->net_idx, addr);
+
+ for (cred = NULL, i = 0; i < ARRAY_SIZE(friend_cred); i++) {
+ if ((friend_cred[i].addr == BT_MESH_ADDR_UNASSIGNED) ||
+ (friend_cred[i].addr == addr &&
+ friend_cred[i].net_idx == sub->net_idx)) {
+ cred = &friend_cred[i];
+ break;
+ }
+ }
+
+ if (!cred) {
+ BT_WARN("No free friend credential slots");
+ return NULL;
+ }
+
+ cred->net_idx = sub->net_idx;
+ cred->addr = addr;
+ cred->lpn_counter = lpn_counter;
+ cred->frnd_counter = frnd_counter;
+
+ err = friend_cred_set(cred, 0, sub->keys[0].net);
+ if (err) {
+ friend_cred_clear(cred);
+ return NULL;
+ }
+
+ if (sub->kr_flag) {
+ err = friend_cred_set(cred, 1, sub->keys[1].net);
+ if (err) {
+ friend_cred_clear(cred);
+ return NULL;
+ }
+ }
+
+ return cred;
+}
+
+void friend_cred_clear(struct friend_cred *cred)
+{
+ cred->net_idx = BT_MESH_KEY_UNUSED;
+ cred->addr = BT_MESH_ADDR_UNASSIGNED;
+ cred->lpn_counter = 0;
+ cred->frnd_counter = 0;
+ memset(cred->cred, 0, sizeof(cred->cred));
+}
+
+int friend_cred_del(u16_t net_idx, u16_t addr)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(friend_cred); i++) {
+ struct friend_cred *cred = &friend_cred[i];
+
+ if (cred->addr == addr && cred->net_idx == net_idx) {
+ friend_cred_clear(cred);
+ return 0;
+ }
+ }
+
+ return -ENOENT;
+}
+
+int friend_cred_get(struct bt_mesh_subnet *sub, u16_t addr, u8_t *nid,
+ const u8_t **enc, const u8_t **priv)
+{
+ int i;
+
+ BT_DBG("net_idx 0x%04x addr 0x%04x", sub->net_idx, addr);
+
+ for (i = 0; i < ARRAY_SIZE(friend_cred); i++) {
+ struct friend_cred *cred = &friend_cred[i];
+
+ if (cred->net_idx != sub->net_idx) {
+ continue;
+ }
+
+ if (addr != BT_MESH_ADDR_UNASSIGNED && cred->addr != addr) {
+ continue;
+ }
+
+ if (nid) {
+ *nid = cred->cred[sub->kr_flag].nid;
+ }
+
+ if (enc) {
+ *enc = cred->cred[sub->kr_flag].enc;
+ }
+
+ if (priv) {
+ *priv = cred->cred[sub->kr_flag].privacy;
+ }
+
+ return 0;
+ }
+
+ return -ENOENT;
+}
+#else
+int friend_cred_get(struct bt_mesh_subnet *sub, u16_t addr, u8_t *nid,
+ const u8_t **enc, const u8_t **priv)
+{
+ return -ENOENT;
+}
+#endif /* FRIEND || LOW_POWER */
+
+u8_t bt_mesh_net_flags(struct bt_mesh_subnet *sub)
+{
+ u8_t flags = 0x00;
+
+ if (sub && sub->kr_flag) {
+ flags |= BT_MESH_NET_FLAG_KR;
+ }
+
+ if (bt_mesh.iv_update) {
+ flags |= BT_MESH_NET_FLAG_IVU;
+ }
+
+ return flags;
+}
+
+int bt_mesh_net_beacon_update(struct bt_mesh_subnet *sub)
+{
+ u8_t flags = bt_mesh_net_flags(sub);
+ struct bt_mesh_subnet_keys *keys;
+
+ if (sub->kr_flag) {
+ BT_DBG("NetIndex %u Using new key", sub->net_idx);
+ keys = &sub->keys[1];
+ } else {
+ BT_DBG("NetIndex %u Using current key", sub->net_idx);
+ keys = &sub->keys[0];
+ }
+
+ BT_DBG("flags 0x%02x, IVI 0x%08x", flags, bt_mesh.iv_index);
+
+ return bt_mesh_beacon_auth(keys->beacon, flags, keys->net_id,
+ bt_mesh.iv_index, sub->auth);
+}
+
+int bt_mesh_net_create(u16_t idx, u8_t flags, const u8_t key[16],
+ u32_t iv_index)
+{
+ struct bt_mesh_subnet *sub;
+ int err;
+
+ BT_DBG("idx %u flags 0x%02x iv_index %u", idx, flags, iv_index);
+
+ BT_DBG("NetKey %s", bt_hex(key, 16));
+
+ if (bt_mesh.valid) {
+ return -EALREADY;
+ }
+
+ memset(msg_cache, 0, sizeof(msg_cache));
+ msg_cache_next = 0;
+
+ sub = &bt_mesh.sub[0];
+
+ sub->kr_flag = BT_MESH_KEY_REFRESH(flags);
+ if (sub->kr_flag) {
+ err = bt_mesh_net_keys_create(&sub->keys[1], key);
+ if (err) {
+ return -EIO;
+ }
+
+ sub->kr_phase = BT_MESH_KR_PHASE_2;
+ } else {
+ err = bt_mesh_net_keys_create(&sub->keys[0], key);
+ if (err) {
+ return -EIO;
+ }
+ }
+
+ bt_mesh.valid = 1;
+ sub->net_idx = idx;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ sub->node_id = BT_MESH_NODE_IDENTITY_RUNNING;
+ sub->node_id_start = k_uptime_get_32();
+ } else {
+ sub->node_id = BT_MESH_NODE_IDENTITY_NOT_SUPPORTED;
+ }
+
+ bt_mesh.iv_index = iv_index;
+ bt_mesh.iv_update = BT_MESH_IV_UPDATE(flags);
+
+ /* Set initial IV Update procedure state time-stamp */
+ bt_mesh.last_update = k_uptime_get();
+
+ /* Make sure valid beacon data has been sent */
+ bt_mesh_net_beacon_update(sub);
+
+ return 0;
+}
+
+void bt_mesh_net_revoke_keys(struct bt_mesh_subnet *sub)
+{
+ int i;
+
+ BT_DBG("idx 0x%04x", sub->net_idx);
+
+ memcpy(&sub->keys[0], &sub->keys[1], sizeof(sub->keys[0]));
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ struct bt_mesh_app_key *key = &bt_mesh.app_keys[i];
+
+ if (key->net_idx != sub->net_idx || !key->updated) {
+ continue;
+ }
+
+ memcpy(&key->keys[0], &key->keys[1], sizeof(key->keys[0]));
+ key->updated = false;
+ }
+}
+
+bool bt_mesh_kr_update(struct bt_mesh_subnet *sub, u8_t new_kr, bool new_key)
+{
+ if (new_kr != sub->kr_flag && sub->kr_phase == BT_MESH_KR_NORMAL) {
+ BT_WARN("KR change in normal operation. Is it a case of blacklisting?");
+ return false;
+ }
+
+ sub->kr_flag = new_kr;
+
+ if (sub->kr_flag) {
+ if (sub->kr_phase == BT_MESH_KR_PHASE_1) {
+ BT_DBG("Phase 1 -> Phase 2");
+ sub->kr_phase = BT_MESH_KR_PHASE_2;
+ return true;
+ }
+ } else {
+ switch (sub->kr_phase) {
+ case BT_MESH_KR_PHASE_1:
+ if (!new_key) {
+ /* Ignore */
+ break;
+ }
+ /* Upon receiving a Secure Network beacon with the KR flag set
+ * to 0 using the new NetKey in Phase 1, the node shall
+ * immediately transition to Phase 3, which effectively skips
+ * Phase 2.
+ *
+ * Intentional fall-through.
+ */
+ case BT_MESH_KR_PHASE_2:
+ BT_DBG("KR Phase 0x%02x -> Normal", sub->kr_phase);
+ bt_mesh_net_revoke_keys(sub);
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER) ||
+ IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ friend_cred_refresh(sub->net_idx);
+ }
+ sub->kr_phase = BT_MESH_KR_NORMAL;
+ return true;
+ }
+ }
+
+ return false;
+}
+
+void bt_mesh_rpl_reset(void)
+{
+ int i;
+
+ /* Discard "old old" IV Index entries from RPL and flag
+ * any other ones (which are valid) as old.
+ */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.rpl); i++) {
+ struct bt_mesh_rpl *rpl = &bt_mesh.rpl[i];
+
+ if (rpl->src) {
+ if (rpl->old_iv) {
+ memset(rpl, 0, sizeof(*rpl));
+ } else {
+ rpl->old_iv = true;
+ }
+ }
+ }
+}
+
+#if defined(CONFIG_BT_MESH_IV_UPDATE_TEST)
+void bt_mesh_iv_update_test(bool enable)
+{
+ bt_mesh.ivu_test = enable;
+}
+
+bool bt_mesh_iv_update(void)
+{
+ if (!bt_mesh_is_provisioned()) {
+ BT_ERR("Not yet provisioned");
+ return false;
+ }
+
+ if (bt_mesh.iv_update) {
+ bt_mesh_net_iv_update(bt_mesh.iv_index, false);
+ } else {
+ bt_mesh_net_iv_update(bt_mesh.iv_index + 1, true);
+ }
+
+ bt_mesh_net_sec_update(NULL);
+
+ return bt_mesh.iv_update;
+}
+#endif /* CONFIG_BT_MESH_IV_UPDATE_TEST */
+
+/* Used for sending immediate beacons to Friend queues and GATT clients */
+void bt_mesh_net_sec_update(struct bt_mesh_subnet *sub)
+{
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ bt_mesh_friend_sec_update(sub ? sub->net_idx : BT_MESH_KEY_ANY);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_ENABLED) {
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_proxy_beacon_send(sub);
+#endif
+ }
+}
+
+bool bt_mesh_net_iv_update(u32_t iv_index, bool iv_update)
+{
+ int i;
+
+ if (bt_mesh.iv_update) {
+ /* It is currently in IV Update mode */
+
+ if (iv_index != bt_mesh.iv_index) {
+ BT_WARN("IV Index mismatch: 0x%08x != 0x%08x",
+ iv_index, bt_mesh.iv_index);
+ return false;
+ }
+
+ if (iv_update) {
+ /* Nothing to do */
+ BT_DBG("Already in IV Update in Progress state");
+ return false;
+ }
+ } else {
+ /* It is currently in Normal mode */
+
+ if (iv_index == bt_mesh.iv_index) {
+ BT_DBG("Same IV Index in normal mode");
+ return false;
+ }
+
+ if (iv_index < bt_mesh.iv_index ||
+ iv_index > bt_mesh.iv_index + 42) {
+ BT_ERR("IV Index out of sync: 0x%08x != 0x%08x",
+ iv_index, bt_mesh.iv_index);
+ return false;
+ }
+
+ if (iv_index > bt_mesh.iv_index + 1) {
+ BT_WARN("Performing IV Index Recovery");
+ memset(bt_mesh.rpl, 0, sizeof(bt_mesh.rpl));
+ bt_mesh.iv_index = iv_index;
+ bt_mesh.seq = 0;
+ goto do_update;
+ }
+
+ if (iv_index == bt_mesh.iv_index + 1 && !iv_update) {
+ BT_WARN("Ignoring new index in normal mode");
+ return false;
+ }
+
+ if (!iv_update) {
+ /* Nothing to do */
+ BT_DBG("Already in Normal state");
+ return false;
+ }
+
+ if (iv_index != bt_mesh.iv_index + 1) {
+ BT_WARN("Wrong new IV Index: 0x%08x != 0x%08x + 1",
+ iv_index, bt_mesh.iv_index);
+ return false;
+ }
+ }
+
+ if (!IS_ENABLED(CONFIG_BT_MESH_IV_UPDATE_TEST) || !bt_mesh.ivu_test) {
+ s64_t delta = k_uptime_get() - bt_mesh.last_update;
+
+ if (delta < K_HOURS(96)) {
+ BT_WARN("IV Update before minimum duration");
+ return false;
+ }
+ }
+
+ /* Defer change to Normal Operation if there are pending ACKs */
+ if (!iv_update && bt_mesh_tx_in_progress()) {
+ BT_WARN("IV Update deferred because of pending transfer");
+ bt_mesh.pending_update = 1;
+ return false;
+ }
+
+do_update:
+ bt_mesh.iv_update = iv_update;
+
+ if (bt_mesh.iv_update) {
+ bt_mesh.iv_index = iv_index;
+ BT_DBG("IV Update state entered. New index 0x%08x",
+ bt_mesh.iv_index);
+
+ bt_mesh_rpl_reset();
+
+ k_delayed_work_submit(&bt_mesh.ivu_complete,
+ IV_UPDATE_TIMEOUT);
+ } else {
+ BT_DBG("Normal mode entered");
+ bt_mesh.seq = 0;
+ k_delayed_work_cancel(&bt_mesh.ivu_complete);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_seq();
+ }
+
+ }
+
+ /* Store time-stamp of the IV procedure state change */
+ bt_mesh.last_update = k_uptime_get();
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ if (bt_mesh.sub[i].net_idx != BT_MESH_KEY_UNUSED) {
+ bt_mesh_net_beacon_update(&bt_mesh.sub[i]);
+ }
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ bt_mesh_store_iv();
+ }
+
+ return true;
+}
+
+u32_t bt_mesh_next_seq(void)
+{
+ u32_t seq = bt_mesh.seq++;
+
+ if (IS_ENABLED(CONFIG_BT_SETTINGS)) {
+ bt_mesh_store_seq();
+ }
+
+ return seq;
+}
+
+int bt_mesh_net_resend(struct bt_mesh_subnet *sub, struct net_buf *buf,
+ bool new_key, const struct bt_mesh_send_cb *cb,
+ void *cb_data)
+{
+ const u8_t *enc, *priv;
+ u32_t seq;
+ int err;
+
+ BT_DBG("net_idx 0x%04x new_key %u len %u", sub->net_idx, new_key,
+ buf->len);
+
+ enc = sub->keys[new_key].enc;
+ priv = sub->keys[new_key].privacy;
+
+ err = bt_mesh_net_obfuscate(buf->data, BT_MESH_NET_IVI_TX, priv);
+ if (err) {
+ BT_ERR("deobfuscate failed (err %d)", err);
+ return err;
+ }
+
+ err = bt_mesh_net_decrypt(enc, &buf->b, BT_MESH_NET_IVI_TX, false);
+ if (err) {
+ BT_ERR("decrypt failed (err %d)", err);
+ return err;
+ }
+
+ /* Update with a new sequence number */
+ seq = bt_mesh_next_seq();
+ buf->data[2] = seq >> 16;
+ buf->data[3] = seq >> 8;
+ buf->data[4] = seq;
+
+ err = bt_mesh_net_encrypt(enc, &buf->b, BT_MESH_NET_IVI_TX, false);
+ if (err) {
+ BT_ERR("encrypt failed (err %d)", err);
+ return err;
+ }
+
+ err = bt_mesh_net_obfuscate(buf->data, BT_MESH_NET_IVI_TX, priv);
+ if (err) {
+ BT_ERR("obfuscate failed (err %d)", err);
+ return err;
+ }
+
+ bt_mesh_adv_send(buf, cb, cb_data);
+
+ if (!bt_mesh.iv_update && bt_mesh.seq > IV_UPDATE_SEQ_LIMIT) {
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_beacon_ivu_initiator(true);
+#endif
+ bt_mesh_net_iv_update(bt_mesh.iv_index + 1, true);
+ bt_mesh_net_sec_update(NULL);
+ }
+
+ return 0;
+}
+
+static void bt_mesh_net_local(struct k_work *work)
+{
+ struct net_buf *buf;
+
+ while ((buf = net_buf_slist_get(&bt_mesh.local_queue))) {
+ BT_DBG("len %u: %s", buf->len, bt_hex(buf->data, buf->len));
+ bt_mesh_net_recv(&buf->b, 0, BT_MESH_NET_IF_LOCAL);
+ net_buf_unref(buf);
+ }
+}
+
+int bt_mesh_net_encode(struct bt_mesh_net_tx *tx, struct net_buf_simple *buf,
+ bool proxy)
+{
+ const bool ctl = (tx->ctx->app_idx == BT_MESH_KEY_UNUSED);
+ u32_t seq_val;
+ u8_t nid;
+ const u8_t *enc, *priv;
+ u8_t *seq;
+ int err;
+
+ if (ctl && net_buf_simple_tailroom(buf) < 8) {
+ BT_ERR("Insufficient MIC space for CTL PDU");
+ return -EINVAL;
+ } else if (net_buf_simple_tailroom(buf) < 4) {
+ BT_ERR("Insufficient MIC space for PDU");
+ return -EINVAL;
+ }
+
+ BT_DBG("src 0x%04x dst 0x%04x ctl %u seq 0x%06x",
+ tx->src, tx->ctx->addr, ctl, bt_mesh.seq);
+
+ net_buf_simple_push_be16(buf, tx->ctx->addr);
+ net_buf_simple_push_be16(buf, tx->src);
+
+ seq = net_buf_simple_push(buf, 3);
+ seq_val = bt_mesh_next_seq();
+ seq[0] = seq_val >> 16;
+ seq[1] = seq_val >> 8;
+ seq[2] = seq_val;
+
+ if (ctl) {
+ net_buf_simple_push_u8(buf, tx->ctx->send_ttl | 0x80);
+ } else {
+ net_buf_simple_push_u8(buf, tx->ctx->send_ttl);
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER) && tx->friend_cred) {
+ if (friend_cred_get(tx->sub, BT_MESH_ADDR_UNASSIGNED,
+ &nid, &enc, &priv)) {
+ BT_WARN("Falling back to master credentials");
+
+ tx->friend_cred = 0;
+
+ nid = tx->sub->keys[tx->sub->kr_flag].nid;
+ enc = tx->sub->keys[tx->sub->kr_flag].enc;
+ priv = tx->sub->keys[tx->sub->kr_flag].privacy;
+ }
+ } else {
+ tx->friend_cred = 0;
+ nid = tx->sub->keys[tx->sub->kr_flag].nid;
+ enc = tx->sub->keys[tx->sub->kr_flag].enc;
+ priv = tx->sub->keys[tx->sub->kr_flag].privacy;
+ }
+
+ net_buf_simple_push_u8(buf, (nid | (BT_MESH_NET_IVI_TX & 1) << 7));
+
+ err = bt_mesh_net_encrypt(enc, buf, BT_MESH_NET_IVI_TX, proxy);
+ if (err) {
+ return err;
+ }
+
+ return bt_mesh_net_obfuscate(buf->data, BT_MESH_NET_IVI_TX, priv);
+}
+
+int bt_mesh_net_send(struct bt_mesh_net_tx *tx, struct net_buf *buf,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ int err;
+
+ BT_DBG("src 0x%04x dst 0x%04x len %u headroom %u tailroom %u",
+ tx->src, tx->ctx->addr, buf->len, net_buf_headroom(buf),
+ net_buf_tailroom(buf));
+ BT_DBG("Payload len %u: %s", buf->len, bt_hex(buf->data, buf->len));
+ BT_DBG("Seq 0x%06x", bt_mesh.seq);
+
+ if (tx->ctx->send_ttl == BT_MESH_TTL_DEFAULT) {
+ tx->ctx->send_ttl = bt_mesh_default_ttl_get();
+ }
+
+ err = bt_mesh_net_encode(tx, &buf->b, false);
+ if (err) {
+ goto done;
+ }
+
+ /* Deliver to GATT Proxy Clients if necessary. Mesh spec 3.4.5.2:
+ * "The output filter of the interface connected to advertising or
+ * GATT bearers shall drop all messages with TTL value set to 1."
+ */
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ tx->ctx->send_ttl != 1) {
+ if (bt_mesh_proxy_relay(&buf->b, tx->ctx->addr) &&
+ BT_MESH_ADDR_IS_UNICAST(tx->ctx->addr)) {
+ /* Notify completion if this only went
+ * through the Mesh Proxy.
+ */
+ if (cb) {
+ if (cb->start) {
+ cb->start(0, 0, cb_data);
+ }
+
+ if (cb->end) {
+ cb->end(0, cb_data);
+ }
+ }
+
+ err = 0;
+ goto done;
+ }
+ }
+ }
+#endif
+
+ /* Deliver to local network interface if necessary */
+ if (bt_mesh_fixed_group_match(tx->ctx->addr) ||
+ bt_mesh_elem_find(tx->ctx->addr)) {
+ if (cb && cb->start) {
+ cb->start(0, 0, cb_data);
+ }
+ net_buf_slist_put(&bt_mesh.local_queue, net_buf_ref(buf));
+ if (cb && cb->end) {
+ cb->end(0, cb_data);
+ }
+ k_work_submit(&bt_mesh.local_work);
+ } else if (tx->ctx->send_ttl != 1) {
+ /* Deliver to the advertising network interface. Mesh spec
+ * 3.4.5.2: "The output filter of the interface connected to
+ * advertising or GATT bearers shall drop all messages with
+ * TTL value set to 1."
+ */
+ bt_mesh_adv_send(buf, cb, cb_data);
+ }
+
+done:
+ net_buf_unref(buf);
+ return err;
+}
+
+static bool auth_match(struct bt_mesh_subnet_keys *keys,
+ const u8_t net_id[8], u8_t flags,
+ u32_t iv_index, const u8_t auth[8])
+{
+ u8_t net_auth[8];
+
+ if (memcmp(net_id, keys->net_id, 8)) {
+ return false;
+ }
+
+ bt_mesh_beacon_auth(keys->beacon, flags, keys->net_id, iv_index,
+ net_auth);
+
+ if (memcmp(auth, net_auth, 8)) {
+ BT_WARN("Authentication Value %s != %s",
+ bt_hex(auth, 8), bt_hex(net_auth, 8));
+ return false;
+ }
+
+ return true;
+}
+
+struct bt_mesh_subnet *bt_mesh_subnet_find(const u8_t net_id[8], u8_t flags,
+ u32_t iv_index, const u8_t auth[8],
+ bool *new_key)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ if (auth_match(&sub->keys[0], net_id, flags, iv_index, auth)) {
+ *new_key = false;
+ return sub;
+ }
+
+ if (sub->kr_phase == BT_MESH_KR_NORMAL) {
+ continue;
+ }
+
+ if (auth_match(&sub->keys[1], net_id, flags, iv_index, auth)) {
+ *new_key = true;
+ return sub;
+ }
+ }
+
+ return NULL;
+}
+
+static int net_decrypt(struct bt_mesh_subnet *sub, const u8_t *enc,
+ const u8_t *priv, const u8_t *data,
+ size_t data_len, struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ BT_DBG("NID 0x%02x net_idx 0x%04x", NID(data), sub->net_idx);
+ BT_DBG("IVI %u net->iv_index 0x%08x", IVI(data), bt_mesh.iv_index);
+
+ rx->old_iv = (IVI(data) != (bt_mesh.iv_index & 0x01));
+
+ net_buf_simple_init(buf, 0);
+ memcpy(net_buf_simple_add(buf, data_len), data, data_len);
+
+ if (bt_mesh_net_obfuscate(buf->data, BT_MESH_NET_IVI_RX(rx), priv)) {
+ return -ENOENT;
+ }
+
+ /* TODO: For provisioner, when a device is re-provisioned and start to
+ * send the same message(e.g. cfg_appkey_add), the status message is easy
+ * to be filtered here. So when a device is re-provisioned, the related
+ * msg_cache should be cleared. Will do it later.
+ */
+ if (rx->net_if == BT_MESH_NET_IF_ADV && msg_cache_match(rx, buf)) {
+ BT_WARN("Duplicate found in Network Message Cache");
+ return -EALREADY;
+ }
+
+ rx->ctx.addr = SRC(buf->data);
+ if (!BT_MESH_ADDR_IS_UNICAST(rx->ctx.addr)) {
+ BT_WARN("Ignoring non-unicast src addr 0x%04x", rx->ctx.addr);
+ return -EINVAL;
+ }
+
+ BT_DBG("src 0x%04x", rx->ctx.addr);
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_PROXY) &&
+ rx->net_if == BT_MESH_NET_IF_PROXY_CFG) {
+ return bt_mesh_net_decrypt(enc, buf, BT_MESH_NET_IVI_RX(rx),
+ true);
+ }
+ }
+#endif
+
+ return bt_mesh_net_decrypt(enc, buf, BT_MESH_NET_IVI_RX(rx), false);
+}
+
+#if (defined(CONFIG_BT_MESH_LOW_POWER) || \
+ defined(CONFIG_BT_MESH_FRIEND))
+static int friend_decrypt(struct bt_mesh_subnet *sub, const u8_t *data,
+ size_t data_len, struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ int i;
+
+ BT_DBG("NID 0x%02x net_idx 0x%04x", NID(data), sub->net_idx);
+
+ for (i = 0; i < ARRAY_SIZE(friend_cred); i++) {
+ struct friend_cred *cred = &friend_cred[i];
+
+ if (cred->net_idx != sub->net_idx) {
+ continue;
+ }
+
+ if (NID(data) == cred->cred[0].nid &&
+ !net_decrypt(sub, cred->cred[0].enc, cred->cred[0].privacy,
+ data, data_len, rx, buf)) {
+ return 0;
+ }
+
+ if (sub->kr_phase == BT_MESH_KR_NORMAL) {
+ continue;
+ }
+
+ if (NID(data) == cred->cred[1].nid &&
+ !net_decrypt(sub, cred->cred[1].enc, cred->cred[1].privacy,
+ data, data_len, rx, buf)) {
+ rx->new_key = 1;
+ return 0;
+ }
+ }
+
+ return -ENOENT;
+}
+#endif
+
+static bool net_find_and_decrypt(const u8_t *data, size_t data_len,
+ struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ u32_t array_size = 0;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (!bt_mesh_is_provisioner_en()) {
+ array_size = ARRAY_SIZE(bt_mesh.sub);
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ array_size = ARRAY_SIZE(bt_mesh.p_sub);
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ array_size = ARRAY_SIZE(bt_mesh.sub);
+ if (bt_mesh_is_provisioner_en()) {
+ array_size += ARRAY_SIZE(bt_mesh.p_sub);
+ }
+#endif
+
+ if (!array_size) {
+ BT_ERR("%s: Get sub array_size fail", __func__);
+ return false;
+ }
+
+ for (i = 0; i < array_size; i++) {
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (!bt_mesh_is_provisioner_en()) {
+ sub = &bt_mesh.sub[i];
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ sub = bt_mesh.p_sub[i];
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (i < ARRAY_SIZE(bt_mesh.sub)) {
+ sub = &bt_mesh.sub[i];
+ } else {
+ sub = bt_mesh.p_sub[i - ARRAY_SIZE(bt_mesh.sub)];
+ }
+#endif
+
+ if (!sub) {
+ BT_DBG("%s: get NULL sub", __func__);
+ continue;
+ }
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+#if (defined(CONFIG_BT_MESH_LOW_POWER) || \
+ defined(CONFIG_BT_MESH_FRIEND))
+ if (!friend_decrypt(sub, data, data_len, rx, buf)) {
+ rx->friend_cred = 1;
+ rx->ctx.net_idx = sub->net_idx;
+ rx->sub = sub;
+ return true;
+ }
+#endif
+ }
+#endif /* CONFIG_BT_MESH_NODE */
+
+ if (NID(data) == sub->keys[0].nid &&
+ !net_decrypt(sub, sub->keys[0].enc, sub->keys[0].privacy,
+ data, data_len, rx, buf)) {
+ rx->ctx.net_idx = sub->net_idx;
+ rx->sub = sub;
+ return true;
+ }
+
+ if (sub->kr_phase == BT_MESH_KR_NORMAL) {
+ continue;
+ }
+
+ if (NID(data) == sub->keys[1].nid &&
+ !net_decrypt(sub, sub->keys[1].enc, sub->keys[1].privacy,
+ data, data_len, rx, buf)) {
+ rx->new_key = 1;
+ rx->ctx.net_idx = sub->net_idx;
+ rx->sub = sub;
+ return true;
+ }
+ }
+
+ return false;
+}
+
+/* Relaying from advertising to the advertising bearer should only happen
+ * if the Relay state is set to enabled. Locally originated packets always
+ * get sent to the advertising bearer. If the packet came in through GATT,
+ * then it should only be relayed if the GATT Proxy state is enabled.
+ */
+#if CONFIG_BT_MESH_NODE
+
+static bool relay_to_adv(enum bt_mesh_net_if net_if)
+{
+ switch (net_if) {
+ case BT_MESH_NET_IF_LOCAL:
+ return true;
+ case BT_MESH_NET_IF_ADV:
+ return (bt_mesh_relay_get() == BT_MESH_RELAY_ENABLED);
+ case BT_MESH_NET_IF_PROXY:
+ return (bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_ENABLED);
+ default:
+ return false;
+ }
+}
+
+static void bt_mesh_net_relay(struct net_buf_simple *sbuf,
+ struct bt_mesh_net_rx *rx)
+{
+ const u8_t *enc, *priv;
+ struct net_buf *buf;
+ u8_t nid, transmit;
+
+ if (rx->net_if == BT_MESH_NET_IF_LOCAL) {
+ /* Locally originated PDUs with TTL=1 will only be delivered
+ * to local elements as per Mesh Profile 1.0 section 3.4.5.2:
+ * "The output filter of the interface connected to
+ * advertising or GATT bearers shall drop all messages with
+ * TTL value set to 1."
+ */
+ if (rx->ctx.recv_ttl == 1) {
+ return;
+ }
+ } else {
+ if (rx->ctx.recv_ttl <= 1) {
+ return;
+ }
+ }
+
+ if (rx->net_if == BT_MESH_NET_IF_ADV &&
+ bt_mesh_relay_get() != BT_MESH_RELAY_ENABLED &&
+ bt_mesh_gatt_proxy_get() != BT_MESH_GATT_PROXY_ENABLED) {
+ return;
+ }
+
+ BT_DBG("TTL %u CTL %u dst 0x%04x", rx->ctx.recv_ttl, rx->ctl, rx->dst);
+
+ /* The Relay Retransmit state is only applied to adv-adv relaying.
+ * Anything else (like GATT to adv, or locally originated packets)
+ * use the Network Transmit state.
+ */
+ if (rx->net_if == BT_MESH_NET_IF_ADV) {
+ transmit = bt_mesh_relay_retransmit_get();
+ } else {
+ transmit = bt_mesh_net_transmit_get();
+ }
+
+ buf = bt_mesh_adv_create(BT_MESH_ADV_DATA,
+ BT_MESH_TRANSMIT_COUNT(transmit),
+ BT_MESH_TRANSMIT_INT(transmit), K_NO_WAIT);
+ if (!buf) {
+ BT_ERR("Out of relay buffers");
+ return;
+ }
+
+ /* Only decrement TTL for non-locally originated packets */
+ if (rx->net_if != BT_MESH_NET_IF_LOCAL) {
+ /* Leave CTL bit intact */
+ sbuf->data[1] &= 0x80;
+ sbuf->data[1] |= rx->ctx.recv_ttl - 1;
+ }
+
+ net_buf_add_mem(buf, sbuf->data, sbuf->len);
+
+ enc = rx->sub->keys[rx->sub->kr_flag].enc;
+ priv = rx->sub->keys[rx->sub->kr_flag].privacy;
+ nid = rx->sub->keys[rx->sub->kr_flag].nid;
+
+ BT_DBG("Relaying packet. TTL is now %u", TTL(buf->data));
+
+ /* Update NID if RX or RX was with friend credentials */
+ if (rx->friend_cred) {
+ buf->data[0] &= 0x80; /* Clear everything except IVI */
+ buf->data[0] |= nid;
+ }
+
+ /* It gets re-encrypted and obfuscated using the received IVI rather than
+ * the normal TX IVI (which may be different) since the transport
+ * layer nonce includes the IVI.
+ */
+ if (bt_mesh_net_encrypt(enc, &buf->b, BT_MESH_NET_IVI_RX(rx), false)) {
+ BT_ERR("Re-encrypting failed");
+ goto done;
+ }
+
+ if (bt_mesh_net_obfuscate(buf->data, BT_MESH_NET_IVI_RX(rx), priv)) {
+ BT_ERR("Re-obfuscating failed");
+ goto done;
+ }
+
+ /* Sending to GATT bearer should only occur if GATT Proxy
+ * is enabled or the message originates from a local node.
+ */
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ (bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_ENABLED ||
+ rx->net_if == BT_MESH_NET_IF_LOCAL)) {
+ if (bt_mesh_proxy_relay(&buf->b, rx->dst) &&
+ BT_MESH_ADDR_IS_UNICAST(rx->dst)) {
+ goto done;
+ }
+ }
+
+ if (relay_to_adv(rx->net_if)) {
+ bt_mesh_adv_send(buf, NULL, NULL);
+ }
+
+done:
+ net_buf_unref(buf);
+}
+
+#endif /* CONFIG_BT_MESH_NODE */
+
+int bt_mesh_net_decode(struct net_buf_simple *data, enum bt_mesh_net_if net_if,
+ struct bt_mesh_net_rx *rx, struct net_buf_simple *buf)
+{
+ if (data->len < BT_MESH_NET_MIN_PDU_LEN) {
+ BT_WARN("Dropping a too short mesh packet (len %u)", data->len);
+ BT_WARN("%s", bt_hex(data->data, data->len));
+ return -EINVAL;
+ }
+
+ if (net_if == BT_MESH_NET_IF_ADV && check_dup(data)) {
+ return -EINVAL;
+ }
+
+ BT_DBG("%u bytes: %s", data->len, bt_hex(data->data, data->len));
+
+ rx->net_if = net_if;
+
+ if (!net_find_and_decrypt(data->data, data->len, rx, buf)) {
+ BT_DBG("Unable to a find matching net for packet");
+ return -ENOENT;
+ }
+
+ /* Initialize AppIdx to a reasonable value */
+ rx->ctx.app_idx = BT_MESH_KEY_UNUSED;
+
+ rx->ctx.recv_ttl = TTL(buf->data);
+
+ /* Default to responding with TTL 0 for non-routed messages */
+ if (rx->ctx.recv_ttl == 0) {
+ rx->ctx.send_ttl = 0;
+ } else {
+ rx->ctx.send_ttl = BT_MESH_TTL_DEFAULT;
+ }
+
+ rx->ctl = CTL(buf->data);
+ rx->seq = SEQ(buf->data);
+ rx->dst = DST(buf->data);
+
+ BT_DBG("Decryption successful. Payload len %u", buf->len);
+
+ if (net_if != BT_MESH_NET_IF_PROXY_CFG &&
+ rx->dst == BT_MESH_ADDR_UNASSIGNED) {
+ BT_ERR("Destination address is unassigned; dropping packet");
+ return -EBADMSG;
+ }
+
+ if (BT_MESH_ADDR_IS_RFU(rx->dst)) {
+ BT_ERR("Destination address is RFU; dropping packet");
+ return -EBADMSG;
+ }
+
+ if (net_if != BT_MESH_NET_IF_LOCAL && bt_mesh_elem_find(rx->ctx.addr)) {
+ BT_DBG("Dropping locally originated packet");
+ return -EBADMSG;
+ }
+
+ BT_DBG("src 0x%04x dst 0x%04x ttl %u", rx->ctx.addr, rx->dst,
+ rx->ctx.recv_ttl);
+ BT_DBG("PDU: %s", bt_hex(buf->data, buf->len));
+
+ return 0;
+}
+
+void bt_mesh_net_recv(struct net_buf_simple *data, s8_t rssi,
+ enum bt_mesh_net_if net_if)
+{
+ struct net_buf_simple *buf = NET_BUF_SIMPLE(29);
+ struct bt_mesh_net_rx rx = { .rssi = rssi };
+ struct net_buf_simple_state state;
+
+ BT_DBG("rssi %d net_if %u", rssi, net_if);
+
+#if CONFIG_BT_MESH_NODE
+ if (!bt_mesh_is_provisioner_en()) {
+ if (!bt_mesh_is_provisioned()) {
+ return;
+ }
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (!bt_mesh_is_provisioner_en()) {
+ BT_WARN("%s: provisioner disabled", __func__);
+ return;
+ }
+ if (!provisioner_get_prov_node_count()) {
+ return;
+ }
+#endif
+
+ if (bt_mesh_net_decode(data, net_if, &rx, buf)) {
+ return;
+ }
+
+ /* Save the state so the buffer can later be relayed */
+ net_buf_simple_save(buf, &state);
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ net_if == BT_MESH_NET_IF_PROXY) {
+ bt_mesh_proxy_addr_add(data, rx.ctx.addr);
+ }
+ }
+#endif
+
+ rx.local_match = (bt_mesh_fixed_group_match(rx.dst) ||
+ bt_mesh_elem_find(rx.dst));
+
+ bt_mesh_trans_recv(buf, &rx);
+
+ /* Relay if this is a group/virtual address, or if the destination
+ * is neither a local element nor an Friend's LPN.
+ */
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (!BT_MESH_ADDR_IS_UNICAST(rx.dst) ||
+ (!rx.local_match && !rx.friend_match)) {
+ net_buf_simple_restore(buf, &state);
+ bt_mesh_net_relay(buf, &rx);
+ }
+ }
+#endif
+}
+
+static void ivu_complete(struct k_work *work)
+{
+ BT_DBG("%s", __func__);
+
+#if CONFIG_BT_MESH_NODE
+ bt_mesh_beacon_ivu_initiator(true);
+#endif
+ bt_mesh_net_iv_update(bt_mesh.iv_index, false);
+}
+
+#if defined(CONFIG_BT_MESH_NODE)
+void bt_mesh_net_start(void)
+{
+ if (bt_mesh_beacon_get() == BT_MESH_BEACON_ENABLED) {
+ bt_mesh_beacon_enable();
+ } else {
+ bt_mesh_beacon_disable();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY) &&
+ bt_mesh_gatt_proxy_get() != BT_MESH_GATT_PROXY_NOT_SUPPORTED) {
+ bt_mesh_proxy_gatt_enable();
+ bt_mesh_adv_update();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER)) {
+ bt_mesh_lpn_init();
+ } else {
+ bt_mesh_scan_enable();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ bt_mesh_friend_init();
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PROV)) {
+ u16_t net_idx = bt_mesh.sub[0].net_idx;
+ u16_t addr = bt_mesh_primary_addr();
+ u8_t flags = (bt_mesh.iv_update << 1) | bt_mesh.sub[0].kr_flag;
+ u32_t iv_index = bt_mesh.iv_index;
+
+ bt_mesh_prov_complete(net_idx, addr, flags, iv_index);
+ }
+}
+#endif
+
+void bt_mesh_net_init(void)
+{
+ k_delayed_work_init(&bt_mesh.ivu_complete, ivu_complete);
+
+ k_work_init(&bt_mesh.local_work, bt_mesh_net_local);
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/net.h b/components/bt/ble_mesh/mesh_core/net.h
new file mode 100644
index 0000000000..897f3775e9
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/net.h
@@ -0,0 +1,391 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef _NET_H_
+#define _NET_H_
+#include "mesh_util.h"
+#include "mesh_kernel.h"
+#include "mesh_access.h"
+
+#define BT_MESH_NET_FLAG_KR BIT(0)
+#define BT_MESH_NET_FLAG_IVU BIT(1)
+
+#define BT_MESH_KR_NORMAL 0x00
+#define BT_MESH_KR_PHASE_1 0x01
+#define BT_MESH_KR_PHASE_2 0x02
+#define BT_MESH_KR_PHASE_3 0x03
+
+#define BT_MESH_IV_UPDATE(flags) ((flags >> 1) & 0x01)
+#define BT_MESH_KEY_REFRESH(flags) (flags & 0x01)
+
+/* Special time-stamp to indicate that we don't know when the last IV
+ * Update happened.
+ */
+#define BT_MESH_NET_IVU_UNKNOWN -1
+
+#if defined(CONFIG_BT_MESH_IV_UPDATE_TEST)
+/* Small test timeout for IV Update Procedure testing */
+#define BT_MESH_NET_IVU_TIMEOUT K_SECONDS(120)
+#else
+/* Maximum time to stay in IV Update mode (96 < time < 144) */
+#define BT_MESH_NET_IVU_TIMEOUT K_HOURS(120)
+#endif /* CONFIG_BT_MESH_IV_UPDATE_TEST */
+
+struct bt_mesh_app_key {
+ u16_t net_idx;
+ u16_t app_idx;
+ bool updated;
+ struct bt_mesh_app_keys {
+ u8_t id;
+ u8_t val[16];
+ } keys[2];
+};
+
+struct bt_mesh_subnet {
+ u32_t beacon_sent; /* Timestamp of last sent beacon */
+ u8_t beacons_last; /* Number of beacons during last
+ * observation window
+ */
+ u8_t beacons_cur; /* Number of beaconds observed during
+ * currently ongoing window.
+ */
+
+ u8_t beacon_cache[21]; /* Cached last authenticated beacon */
+
+ u16_t net_idx; /* NetKeyIndex */
+
+ bool kr_flag; /* Key Refresh Flag */
+ u8_t kr_phase; /* Key Refresh Phase */
+
+ u8_t node_id; /* Node Identity State */
+ u32_t node_id_start; /* Node Identity started timestamp */
+
+ u8_t auth[8]; /* Beacon Authentication Value */
+
+ struct bt_mesh_subnet_keys {
+ u8_t net[16]; /* NetKey */
+ u8_t nid; /* NID */
+ u8_t enc[16]; /* EncKey */
+ u8_t net_id[8]; /* Network ID */
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ u8_t identity[16]; /* IdentityKey */
+#endif
+ u8_t privacy[16]; /* PrivacyKey */
+ u8_t beacon[16]; /* BeaconKey */
+ } keys[2];
+};
+
+struct bt_mesh_rpl {
+ u16_t src;
+ bool old_iv;
+#if defined(CONFIG_BT_MESH_SETTINGS)
+ bool store;
+#endif
+ u32_t seq;
+};
+
+#if defined(CONFIG_BT_MESH_FRIEND)
+#define FRIEND_SEG_RX CONFIG_BT_MESH_FRIEND_SEG_RX
+#define FRIEND_SUB_LIST_SIZE CONFIG_BT_MESH_FRIEND_SUB_LIST_SIZE
+#else
+#define FRIEND_SEG_RX 0
+#define FRIEND_SUB_LIST_SIZE 0
+#endif
+
+struct bt_mesh_friend {
+ u16_t lpn;
+ u8_t recv_delay;
+ u8_t fsn: 1,
+ send_last: 1,
+ pending_req: 1,
+ sec_update: 1,
+ pending_buf: 1,
+ valid: 1,
+ established: 1;
+ s32_t poll_to;
+ u8_t num_elem;
+ u16_t lpn_counter;
+ u16_t counter;
+
+ u16_t net_idx;
+
+ u16_t sub_list[FRIEND_SUB_LIST_SIZE];
+
+ struct k_delayed_work timer;
+
+ struct bt_mesh_friend_seg {
+ sys_slist_t queue;
+ } seg[FRIEND_SEG_RX];
+
+ struct net_buf *last;
+
+ sys_slist_t queue;
+ u32_t queue_size;
+
+ /* Friend Clear Procedure */
+ struct {
+ u32_t start; /* Clear Procedure start */
+ u16_t frnd; /* Previous Friend's address */
+ u16_t repeat_sec; /* Repeat timeout in seconds */
+ struct k_delayed_work timer; /* Repeat timer */
+ } clear;
+};
+
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+#define LPN_GROUPS CONFIG_BT_MESH_LOW_POWER
+#else
+#define LPN_GROUPS 0
+#endif
+
+/* Low Power Node state */
+struct bt_mesh_lpn {
+ enum __packed {
+ BT_MESH_LPN_DISABLED, /* LPN feature is disabled */
+ BT_MESH_LPN_CLEAR, /* Clear in progress */
+ BT_MESH_LPN_TIMER, /* Waiting for auto timer expiry */
+ BT_MESH_LPN_ENABLED, /* LPN enabled, but no Friend */
+ BT_MESH_LPN_REQ_WAIT, /* Wait before scanning for offers */
+ BT_MESH_LPN_WAIT_OFFER, /* Friend Req sent */
+ BT_MESH_LPN_ESTABLISHED, /* Friendship established */
+ BT_MESH_LPN_RECV_DELAY, /* Poll sent, waiting ReceiveDelay */
+ BT_MESH_LPN_WAIT_UPDATE, /* Waiting for Update or message */
+ BT_MESH_LPN_OFFER_RECV, /* Friend offer received */
+ } state;
+
+ /* Transaction Number (used for subscription list) */
+ u8_t xact_next;
+ u8_t xact_pending;
+ u8_t sent_req;
+
+ /* Address of our Friend when we're a LPN. Unassigned if we don't
+ * have a friend yet.
+ */
+ u16_t frnd;
+
+ /* Value from the friend offer */
+ u8_t recv_win;
+
+ u8_t req_attempts; /* Number of Request attempts */
+
+ s32_t poll_timeout;
+
+ u8_t groups_changed: 1, /* Friend Subscription List needs updating */
+ pending_poll: 1, /* Poll to be sent after subscription */
+ disable: 1, /* Disable LPN after clearing */
+ fsn: 1, /* Friend Sequence Number */
+ established: 1, /* Friendship established */
+ clear_success: 1; /* Friend Clear Confirm received */
+
+ /* Friend Queue Size */
+ u8_t queue_size;
+
+ /* LPNCounter */
+ u16_t counter;
+
+ /* Previous Friend of this LPN */
+ u16_t old_friend;
+
+ /* Duration reported for last advertising packet */
+ u16_t adv_duration;
+
+ /* Next LPN related action timer */
+ struct k_delayed_work timer;
+
+ /* Subscribed groups */
+ u16_t groups[LPN_GROUPS];
+
+ /* Bit fields for tracking which groups the Friend knows about */
+ ATOMIC_DEFINE(added, LPN_GROUPS);
+ ATOMIC_DEFINE(pending, LPN_GROUPS);
+ ATOMIC_DEFINE(to_remove, LPN_GROUPS);
+};
+
+/* bt_mesh_net.flags, mainly used for pending storage actions */
+enum {
+ BT_MESH_RPL_PENDING,
+ BT_MESH_KEYS_PENDING,
+ BT_MESH_NET_PENDING,
+ BT_MESH_IV_PENDING,
+ BT_MESH_SEQ_PENDING,
+ BT_MESH_HB_PUB_PENDING,
+ BT_MESH_CFG_PENDING,
+ BT_MESH_MOD_PENDING,
+
+ /* Don't touch - intentionally last */
+ BT_MESH_FLAG_COUNT,
+};
+
+struct bt_mesh_net {
+ u32_t iv_index; /* Current IV Index */
+ u32_t seq: 24, /* Next outgoing sequence number */
+ iv_update: 1, /* 1 if IV Update in Progress */
+ ivu_initiator: 1, /* IV Update initiated by us */
+ ivu_test: 1, /* IV Update test mode */
+ pending_update: 1, /* Update blocked by SDU in progress */
+ valid: 1; /* 0 if unused */
+
+ s64_t last_update; /* Time since last IV Update change */
+
+ ATOMIC_DEFINE(flags, BT_MESH_FLAG_COUNT);
+
+ /* Local network interface */
+ struct k_work local_work;
+ sys_slist_t local_queue;
+
+#if defined(CONFIG_BT_MESH_FRIEND)
+ /* Friend state, unique for each LPN that we're Friends for */
+ struct bt_mesh_friend frnd[CONFIG_BT_MESH_FRIEND_LPN_COUNT];
+#endif
+
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+ struct bt_mesh_lpn lpn; /* Low Power Node state */
+#endif
+
+ /* Timer to transition IV Update in Progress state */
+ struct k_delayed_work ivu_complete;
+
+ u8_t dev_key[16];
+
+ struct bt_mesh_app_key app_keys[CONFIG_BT_MESH_APP_KEY_COUNT];
+
+ struct bt_mesh_subnet sub[CONFIG_BT_MESH_SUBNET_COUNT];
+
+ struct bt_mesh_rpl rpl[CONFIG_BT_MESH_CRPL];
+
+#if CONFIG_BT_MESH_PROVISIONER
+ /* Provisioner and node share arguments from 'iv_index' to 'local_queue' + 'ivu_complete' */
+
+ /* Application keys stored by provisioner */
+ struct bt_mesh_app_key *p_app_keys[CONFIG_BT_MESH_PROVISIONER_APP_KEY_COUNT];
+ /* Next app_idx can be assigned */
+ u16_t p_app_idx_next;
+
+ /* Network keys stored by provisioner */
+ struct bt_mesh_subnet *p_sub[CONFIG_BT_MESH_PROVISIONER_SUBNET_COUNT];
+ /* Next net_idx can be assigned */
+ u16_t p_net_idx_next;
+#endif
+};
+
+/* Network interface */
+enum bt_mesh_net_if {
+ BT_MESH_NET_IF_ADV,
+ BT_MESH_NET_IF_LOCAL,
+ BT_MESH_NET_IF_PROXY,
+ BT_MESH_NET_IF_PROXY_CFG,
+};
+
+/* Decoding context for Network/Transport data */
+struct bt_mesh_net_rx {
+ struct bt_mesh_subnet *sub;
+ struct bt_mesh_msg_ctx ctx;
+ u32_t seq; /* Sequence Number */
+ u16_t dst; /* Destination address */
+ u8_t old_iv: 1, /* iv_index - 1 was used */
+ new_key: 1, /* Data was encrypted with updated key */
+ friend_cred: 1, /* Data was encrypted with friend cred */
+ ctl: 1, /* Network Control */
+ net_if: 2, /* Network interface */
+ local_match: 1, /* Matched a local element */
+ friend_match: 1; /* Matched an LPN we're friends for */
+ s8_t rssi;
+};
+
+/* Encoding context for Network/Transport data */
+struct bt_mesh_net_tx {
+ struct bt_mesh_subnet *sub;
+ struct bt_mesh_msg_ctx *ctx;
+ u16_t src;
+ u8_t xmit;
+ u8_t friend_cred: 1,
+ aszmic: 1,
+ aid: 6;
+};
+
+extern struct bt_mesh_net bt_mesh;
+
+#define BT_MESH_NET_IVI_TX (bt_mesh.iv_index - bt_mesh.iv_update)
+#define BT_MESH_NET_IVI_RX(rx) (bt_mesh.iv_index - (rx)->old_iv)
+
+#define BT_MESH_NET_HDR_LEN 9
+
+int bt_mesh_net_keys_create(struct bt_mesh_subnet_keys *keys,
+ const u8_t key[16]);
+
+int bt_mesh_net_create(u16_t idx, u8_t flags, const u8_t key[16],
+ u32_t iv_index);
+
+u8_t bt_mesh_net_flags(struct bt_mesh_subnet *sub);
+
+bool bt_mesh_kr_update(struct bt_mesh_subnet *sub, u8_t new_kr, bool new_key);
+
+void bt_mesh_net_revoke_keys(struct bt_mesh_subnet *sub);
+
+int bt_mesh_net_beacon_update(struct bt_mesh_subnet *sub);
+
+void bt_mesh_rpl_reset(void);
+
+bool bt_mesh_net_iv_update(u32_t iv_index, bool iv_update);
+
+void bt_mesh_net_sec_update(struct bt_mesh_subnet *sub);
+
+struct bt_mesh_subnet *bt_mesh_subnet_get(u16_t net_idx);
+
+struct bt_mesh_subnet *bt_mesh_subnet_find(const u8_t net_id[8], u8_t flags,
+ u32_t iv_index, const u8_t auth[8],
+ bool *new_key);
+
+int bt_mesh_net_encode(struct bt_mesh_net_tx *tx, struct net_buf_simple *buf,
+ bool proxy);
+
+int bt_mesh_net_send(struct bt_mesh_net_tx *tx, struct net_buf *buf,
+ const struct bt_mesh_send_cb *cb, void *cb_data);
+
+int bt_mesh_net_resend(struct bt_mesh_subnet *sub, struct net_buf *buf,
+ bool new_key, const struct bt_mesh_send_cb *cb,
+ void *cb_data);
+
+int bt_mesh_net_decode(struct net_buf_simple *data, enum bt_mesh_net_if net_if,
+ struct bt_mesh_net_rx *rx, struct net_buf_simple *buf);
+
+void bt_mesh_net_recv(struct net_buf_simple *data, s8_t rssi,
+ enum bt_mesh_net_if net_if);
+
+u32_t bt_mesh_next_seq(void);
+
+void bt_mesh_net_init(void);
+
+void bt_mesh_net_start(void);
+
+/* Friendship Credential Management */
+struct friend_cred {
+ u16_t net_idx;
+ u16_t addr;
+
+ u16_t lpn_counter;
+ u16_t frnd_counter;
+
+ struct {
+ u8_t nid; /* NID */
+ u8_t enc[16]; /* EncKey */
+ u8_t privacy[16]; /* PrivacyKey */
+ } cred[2];
+};
+
+int friend_cred_get(struct bt_mesh_subnet *sub, u16_t addr, u8_t *nid,
+ const u8_t **enc, const u8_t **priv);
+int friend_cred_set(struct friend_cred *cred, u8_t idx, const u8_t net_key[16]);
+void friend_cred_refresh(u16_t net_idx);
+int friend_cred_update(struct bt_mesh_subnet *sub);
+struct friend_cred *friend_cred_create(struct bt_mesh_subnet *sub, u16_t addr,
+ u16_t lpn_counter, u16_t frnd_counter);
+void friend_cred_clear(struct friend_cred *cred);
+int friend_cred_del(u16_t net_idx, u16_t addr);
+
+#endif /* #ifndef _NET_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/prov.c b/components/bt/ble_mesh/mesh_core/prov.c
new file mode 100644
index 0000000000..8247984b0c
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/prov.c
@@ -0,0 +1,1763 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#include
+#include
+
+#include "mesh_util.h"
+
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+
+#include "mesh_main.h"
+#include "mesh_bt_uuid.h"
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_PROV)
+#include "mesh_trace.h"
+
+#include "crypto.h"
+#include "adv.h"
+#include "mesh.h"
+#include "net.h"
+#include "access.h"
+#include "foundation.h"
+#include "proxy.h"
+#include "prov.h"
+
+#if CONFIG_BT_MESH_NODE
+
+/* 3 transmissions, 20ms interval */
+#define PROV_XMIT_COUNT 2
+#define PROV_XMIT_INT 20
+
+#define AUTH_METHOD_NO_OOB 0x00
+#define AUTH_METHOD_STATIC 0x01
+#define AUTH_METHOD_OUTPUT 0x02
+#define AUTH_METHOD_INPUT 0x03
+
+#define OUTPUT_OOB_BLINK 0x00
+#define OUTPUT_OOB_BEEP 0x01
+#define OUTPUT_OOB_VIBRATE 0x02
+#define OUTPUT_OOB_NUMBER 0x03
+#define OUTPUT_OOB_STRING 0x04
+
+#define INPUT_OOB_PUSH 0x00
+#define INPUT_OOB_TWIST 0x01
+#define INPUT_OOB_NUMBER 0x02
+#define INPUT_OOB_STRING 0x03
+
+#define PROV_ERR_NONE 0x00
+#define PROV_ERR_NVAL_PDU 0x01
+#define PROV_ERR_NVAL_FMT 0x02
+#define PROV_ERR_UNEXP_PDU 0x03
+#define PROV_ERR_CFM_FAILED 0x04
+#define PROV_ERR_RESOURCES 0x05
+#define PROV_ERR_DECRYPT 0x06
+#define PROV_ERR_UNEXP_ERR 0x07
+#define PROV_ERR_ADDR 0x08
+
+#define PROV_INVITE 0x00
+#define PROV_CAPABILITIES 0x01
+#define PROV_START 0x02
+#define PROV_PUB_KEY 0x03
+#define PROV_INPUT_COMPLETE 0x04
+#define PROV_CONFIRM 0x05
+#define PROV_RANDOM 0x06
+#define PROV_DATA 0x07
+#define PROV_COMPLETE 0x08
+#define PROV_FAILED 0x09
+
+#define PROV_ALG_P256 0x00
+
+#define GPCF(gpc) (gpc & 0x03)
+#define GPC_START(last_seg) (((last_seg) << 2) | 0x00)
+#define GPC_ACK 0x01
+#define GPC_CONT(seg_id) (((seg_id) << 2) | 0x02)
+#define GPC_CTL(op) (((op) << 2) | 0x03)
+
+#define START_PAYLOAD_MAX 20
+#define CONT_PAYLOAD_MAX 23
+
+#define START_LAST_SEG(gpc) (gpc >> 2)
+#define CONT_SEG_INDEX(gpc) (gpc >> 2)
+
+#define BEARER_CTL(gpc) (gpc >> 2)
+#define LINK_OPEN 0x00
+#define LINK_ACK 0x01
+#define LINK_CLOSE 0x02
+
+#define CLOSE_REASON_SUCCESS 0x00
+#define CLOSE_REASON_TIMEOUT 0x01
+#define CLOSE_REASON_FAILED 0x02
+
+#define XACT_SEG_DATA(_seg) (&link.rx.buf->data[20 + ((_seg - 1) * 23)])
+#define XACT_SEG_RECV(_seg) (link.rx.seg &= ~(1 << (_seg)))
+
+#define XACT_NVAL 0xff
+
+enum {
+ REMOTE_PUB_KEY, /* Remote key has been received */
+ LOCAL_PUB_KEY, /* Local public key is available */
+ OOB_PUB_KEY, /* OOB public key is available */
+ LINK_ACTIVE, /* Link has been opened */
+ HAVE_DHKEY, /* DHKey has been calcualted */
+ SEND_CONFIRM, /* Waiting to send Confirm value */
+ WAIT_NUMBER, /* Waiting for number input from user */
+ WAIT_STRING, /* Waiting for string input from user */
+ TIMEOUT_START, /* Provision timeout timer has started */
+
+ NUM_FLAGS,
+};
+
+struct prov_link {
+ ATOMIC_DEFINE(flags, NUM_FLAGS);
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ struct bt_conn *conn; /* GATT connection */
+#endif
+ u8_t dhkey[32]; /* Calculated DHKey */
+ u8_t expect; /* Next expected PDU */
+
+ bool oob_pk_flag; /* Flag indicates whether using OOB public key */
+
+ u8_t oob_method;
+ u8_t oob_action;
+ u8_t oob_size;
+
+ u8_t conf[16]; /* Remote Confirmation */
+ u8_t rand[16]; /* Local Random */
+ u8_t auth[16]; /* Authentication Value */
+
+ u8_t conf_salt[16]; /* ConfirmationSalt */
+ u8_t conf_key[16]; /* ConfirmationKey */
+ u8_t conf_inputs[145]; /* ConfirmationInputs */
+ u8_t prov_salt[16]; /* Provisioning Salt */
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ u32_t id; /* Link ID */
+
+ struct {
+ u8_t id; /* Transaction ID */
+ u8_t prev_id; /* Previous Transaction ID */
+ u8_t seg; /* Bit-field of unreceived segments */
+ u8_t last_seg; /* Last segment (to check length) */
+ u8_t fcs; /* Expected FCS value */
+ struct net_buf_simple *buf;
+ } rx;
+
+ struct {
+ /* Start timestamp of the transaction */
+ s64_t start;
+
+ /* Transaction id*/
+ u8_t id;
+
+ /* Pending outgoing buffer(s) */
+ struct net_buf *buf[3];
+
+ /* Retransmit timer */
+ struct k_delayed_work retransmit;
+ } tx;
+#endif
+
+ /* Provision timeout timer */
+ struct k_delayed_work timeout;
+};
+
+struct prov_rx {
+ u32_t link_id;
+ u8_t xact_id;
+ u8_t gpc;
+};
+
+#define RETRANSMIT_TIMEOUT K_MSEC(500)
+#define BUF_TIMEOUT K_MSEC(400)
+#if defined(CONFIG_BT_MESH_FAST_PROV)
+#define TRANSACTION_TIMEOUT K_SECONDS(3)
+#define PROVISION_TIMEOUT K_SECONDS(6)
+#else
+#define TRANSACTION_TIMEOUT K_SECONDS(30)
+#define PROVISION_TIMEOUT K_SECONDS(60)
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+#define PROV_BUF_HEADROOM 5
+#else
+#define PROV_BUF_HEADROOM 0
+static struct net_buf_simple *rx_buf = NET_BUF_SIMPLE(65);
+#endif
+
+#define PROV_BUF(len) NET_BUF_SIMPLE(PROV_BUF_HEADROOM + len)
+
+static struct prov_link link;
+
+static const struct bt_mesh_prov *prov;
+
+static void close_link(u8_t err, u8_t reason);
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+static void buf_sent(int err, void *user_data)
+{
+ if (!link.tx.buf[0]) {
+ return;
+ }
+
+ k_delayed_work_submit(&link.tx.retransmit, RETRANSMIT_TIMEOUT);
+}
+
+static struct bt_mesh_send_cb buf_sent_cb = {
+ .end = buf_sent,
+};
+
+static void free_segments(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(link.tx.buf); i++) {
+ struct net_buf *buf = link.tx.buf[i];
+
+ if (!buf) {
+ break;
+ }
+
+ link.tx.buf[i] = NULL;
+ /* Mark as canceled */
+ BT_MESH_ADV(buf)->busy = 0;
+ /** Change by Espressif. Add this to avoid buf->ref is 2 which will
+ * cause lack of buf.
+ */
+ if (buf->ref > 1) {
+ buf->ref = 1;
+ }
+ net_buf_unref(buf);
+ }
+}
+
+static void prov_clear_tx(void)
+{
+ BT_DBG("%s", __func__);
+
+ k_delayed_work_cancel(&link.tx.retransmit);
+
+ free_segments();
+}
+
+static void reset_link(void)
+{
+ prov_clear_tx();
+
+ /* Disable Attention Timer if it was set before */
+ if (link.conf_inputs[0]) {
+ bt_mesh_attention(NULL, 0);
+ }
+
+ if (atomic_test_and_clear_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link.timeout);
+ }
+
+ if (prov->link_close) {
+ prov->link_close(BT_MESH_PROV_ADV);
+ }
+
+ /* Clear everything except the retransmit delayed work config */
+ memset(&link, 0, offsetof(struct prov_link, tx.retransmit));
+
+ link.rx.prev_id = XACT_NVAL;
+
+ if (bt_mesh_pub_key_get()) {
+ atomic_set_bit(link.flags, LOCAL_PUB_KEY);
+ }
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ link.rx.buf = bt_mesh_proxy_get_buf();
+#else
+ net_buf_simple_init(rx_buf, 0);
+ link.rx.buf = rx_buf;
+#endif
+}
+
+static struct net_buf *adv_buf_create(void)
+{
+ struct net_buf *buf;
+
+ buf = bt_mesh_adv_create(BT_MESH_ADV_PROV, PROV_XMIT_COUNT,
+ PROV_XMIT_INT, BUF_TIMEOUT);
+ if (!buf) {
+ BT_ERR("Out of provisioning buffers");
+ return NULL;
+ }
+
+ return buf;
+}
+
+static u8_t pending_ack = XACT_NVAL;
+
+static void ack_complete(u16_t duration, int err, void *user_data)
+{
+ BT_DBG("xact %u complete", (u8_t)pending_ack);
+ pending_ack = XACT_NVAL;
+}
+
+static void gen_prov_ack_send(u8_t xact_id)
+{
+ static const struct bt_mesh_send_cb cb = {
+ .start = ack_complete,
+ };
+ const struct bt_mesh_send_cb *complete;
+ struct net_buf *buf;
+
+ BT_DBG("xact_id %u", xact_id);
+
+ if (pending_ack == xact_id) {
+ BT_DBG("Not sending duplicate ACK");
+ return;
+ }
+
+ buf = adv_buf_create();
+ if (!buf) {
+ return;
+ }
+
+ if (pending_ack == XACT_NVAL) {
+ pending_ack = xact_id;
+ complete = &cb;
+ } else {
+ complete = NULL;
+ }
+
+ net_buf_add_be32(buf, link.id);
+ net_buf_add_u8(buf, xact_id);
+ net_buf_add_u8(buf, GPC_ACK);
+
+ bt_mesh_adv_send(buf, complete, NULL);
+ net_buf_unref(buf);
+}
+
+static void send_reliable(void)
+{
+ int i;
+
+ link.tx.start = k_uptime_get();
+
+ for (i = 0; i < ARRAY_SIZE(link.tx.buf); i++) {
+ struct net_buf *buf = link.tx.buf[i];
+
+ if (!buf) {
+ break;
+ }
+
+ if (i + 1 < ARRAY_SIZE(link.tx.buf) && link.tx.buf[i + 1]) {
+ bt_mesh_adv_send(buf, NULL, NULL);
+ } else {
+ bt_mesh_adv_send(buf, &buf_sent_cb, NULL);
+ }
+ }
+}
+
+static int bearer_ctl_send(u8_t op, void *data, u8_t data_len)
+{
+ struct net_buf *buf;
+
+ BT_DBG("op 0x%02x data_len %u", op, data_len);
+
+ prov_clear_tx();
+
+ buf = adv_buf_create();
+ if (!buf) {
+ return -ENOBUFS;
+ }
+
+ net_buf_add_be32(buf, link.id);
+ /* Transaction ID, always 0 for Bearer messages */
+ net_buf_add_u8(buf, 0x00);
+ net_buf_add_u8(buf, GPC_CTL(op));
+ net_buf_add_mem(buf, data, data_len);
+
+ link.tx.buf[0] = buf;
+ send_reliable();
+
+ return 0;
+}
+
+static u8_t last_seg(u8_t len)
+{
+ if (len <= START_PAYLOAD_MAX) {
+ return 0;
+ }
+
+ len -= START_PAYLOAD_MAX;
+
+ return 1 + (len / CONT_PAYLOAD_MAX);
+}
+
+static inline u8_t next_transaction_id(void)
+{
+ if (link.tx.id != 0 && link.tx.id != 0xFF) {
+ return ++link.tx.id;
+ }
+
+ link.tx.id = 0x80;
+ return link.tx.id;
+}
+
+static int prov_send_adv(struct net_buf_simple *msg)
+{
+ struct net_buf *start, *buf;
+ u8_t seg_len, seg_id;
+ u8_t xact_id;
+ u8_t type;
+
+ BT_DBG("%s, len %u: %s", __func__, msg->len, bt_hex(msg->data, msg->len));
+
+ prov_clear_tx();
+
+ start = adv_buf_create();
+ if (!start) {
+ return -ENOBUFS;
+ }
+
+ xact_id = next_transaction_id();
+ net_buf_add_be32(start, link.id);
+ net_buf_add_u8(start, xact_id);
+
+ net_buf_add_u8(start, GPC_START(last_seg(msg->len)));
+ net_buf_add_be16(start, msg->len);
+ net_buf_add_u8(start, bt_mesh_fcs_calc(msg->data, msg->len));
+
+ link.tx.buf[0] = start;
+ /* Change by Espressif, get message type */
+ type = msg->data[0];
+
+ seg_len = min(msg->len, START_PAYLOAD_MAX);
+ BT_DBG("seg 0 len %u: %s", seg_len, bt_hex(msg->data, seg_len));
+ net_buf_add_mem(start, msg->data, seg_len);
+ net_buf_simple_pull(msg, seg_len);
+
+ buf = start;
+ for (seg_id = 1; msg->len > 0; seg_id++) {
+ if (seg_id >= ARRAY_SIZE(link.tx.buf)) {
+ BT_ERR("Too big message");
+ free_segments();
+ return -E2BIG;
+ }
+
+ buf = adv_buf_create();
+ if (!buf) {
+ free_segments();
+ return -ENOBUFS;
+ }
+
+ link.tx.buf[seg_id] = buf;
+
+ seg_len = min(msg->len, CONT_PAYLOAD_MAX);
+
+ BT_DBG("seg_id %u len %u: %s", seg_id, seg_len,
+ bt_hex(msg->data, seg_len));
+
+ net_buf_add_be32(buf, link.id);
+ net_buf_add_u8(buf, xact_id);
+ net_buf_add_u8(buf, GPC_CONT(seg_id));
+ net_buf_add_mem(buf, msg->data, seg_len);
+ net_buf_simple_pull(msg, seg_len);
+ }
+
+ send_reliable();
+
+ /* Change by Espressif, add provisioning timeout timer operations.
+ * When sending a provisioning PDU successfully, restart the 60s timer.
+ */
+ if (atomic_test_and_clear_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link.timeout);
+ }
+ if (type != PROV_COMPLETE && type != PROV_FAILED) {
+ if (!atomic_test_and_set_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_submit(&link.timeout, PROVISION_TIMEOUT);
+ }
+ }
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+static int prov_send_gatt(struct net_buf_simple *msg)
+{
+ int err = 0;
+
+ if (!link.conn) {
+ return -ENOTCONN;
+ }
+
+ /* Change by Espressif, add provisioning timeout timer operations.
+ * When sending a provisioning PDU successfully, restart the 60s timer.
+ */
+ err = bt_mesh_proxy_send(link.conn, BT_MESH_PROXY_PROV, msg);
+ if (err) {
+ BT_ERR("%s: fail to send provisioning PDU", __func__);
+ return err;
+ }
+
+ if (atomic_test_and_clear_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link.timeout);
+ }
+ if (msg->data[1] != PROV_COMPLETE && msg->data[1] != PROV_FAILED) {
+ if (!atomic_test_and_set_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_submit(&link.timeout, PROVISION_TIMEOUT);
+ }
+ }
+
+ return 0;
+}
+#endif /* CONFIG_BT_MESH_PB_GATT */
+
+static inline int prov_send(struct net_buf_simple *buf)
+{
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (link.conn) {
+ return prov_send_gatt(buf);
+ }
+#endif
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ return prov_send_adv(buf);
+#else
+ return 0;
+#endif
+}
+
+static void prov_buf_init(struct net_buf_simple *buf, u8_t type)
+{
+ net_buf_simple_init(buf, PROV_BUF_HEADROOM);
+ net_buf_simple_add_u8(buf, type);
+}
+
+static void prov_send_fail_msg(u8_t err)
+{
+ struct net_buf_simple *buf = PROV_BUF(2);
+
+ prov_buf_init(buf, PROV_FAILED);
+ net_buf_simple_add_u8(buf, err);
+ prov_send(buf);
+}
+
+static void prov_invite(const u8_t *data)
+{
+ struct net_buf_simple *buf = PROV_BUF(12);
+
+ BT_DBG("Attention Duration: %u seconds", data[0]);
+
+ if (data[0]) {
+ bt_mesh_attention(NULL, data[0]);
+ }
+
+ link.conf_inputs[0] = data[0];
+
+ prov_buf_init(buf, PROV_CAPABILITIES);
+
+ /* Number of Elements supported */
+ net_buf_simple_add_u8(buf, bt_mesh_elem_count());
+
+ /* Supported algorithms - FIPS P-256 Eliptic Curve */
+ net_buf_simple_add_be16(buf, BIT(PROV_ALG_P256));
+
+ /* Public Key Type */
+ net_buf_simple_add_u8(buf, prov->oob_pub_key);
+
+ /* Static OOB Type */
+ net_buf_simple_add_u8(buf, prov->static_val ? BIT(0) : 0x00);
+
+ /* Output OOB Size */
+ net_buf_simple_add_u8(buf, prov->output_size);
+
+ /* Output OOB Action */
+ net_buf_simple_add_be16(buf, prov->output_actions);
+
+ /* Input OOB Size */
+ net_buf_simple_add_u8(buf, prov->input_size);
+
+ /* Input OOB Action */
+ net_buf_simple_add_be16(buf, prov->input_actions);
+
+ memcpy(&link.conf_inputs[1], &buf->data[1], 11);
+
+ if (prov_send(buf)) {
+ BT_ERR("Failed to send capabilities");
+ close_link(PROV_ERR_RESOURCES, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ link.expect = PROV_START;
+}
+
+static void prov_capabilities(const u8_t *data)
+{
+ u16_t algorithms, output_action, input_action;
+
+ BT_DBG("Elements: %u", data[0]);
+
+ algorithms = sys_get_be16(&data[1]);
+ BT_DBG("Algorithms: %u", algorithms);
+
+ BT_DBG("Public Key Type: 0x%02x", data[3]);
+ BT_DBG("Static OOB Type: 0x%02x", data[4]);
+ BT_DBG("Output OOB Size: %u", data[5]);
+
+ output_action = sys_get_be16(&data[6]);
+ BT_DBG("Output OOB Action: 0x%04x", output_action);
+
+ BT_DBG("Input OOB Size: %u", data[8]);
+
+ input_action = sys_get_be16(&data[9]);
+ BT_DBG("Input OOB Action: 0x%04x", input_action);
+}
+
+static bt_mesh_output_action_t output_action(u8_t action)
+{
+ switch (action) {
+ case OUTPUT_OOB_BLINK:
+ return BT_MESH_BLINK;
+ case OUTPUT_OOB_BEEP:
+ return BT_MESH_BEEP;
+ case OUTPUT_OOB_VIBRATE:
+ return BT_MESH_VIBRATE;
+ case OUTPUT_OOB_NUMBER:
+ return BT_MESH_DISPLAY_NUMBER;
+ case OUTPUT_OOB_STRING:
+ return BT_MESH_DISPLAY_STRING;
+ default:
+ return BT_MESH_NO_OUTPUT;
+ }
+}
+
+static bt_mesh_input_action_t input_action(u8_t action)
+{
+ switch (action) {
+ case INPUT_OOB_PUSH:
+ return BT_MESH_PUSH;
+ case INPUT_OOB_TWIST:
+ return BT_MESH_TWIST;
+ case INPUT_OOB_NUMBER:
+ return BT_MESH_ENTER_NUMBER;
+ case INPUT_OOB_STRING:
+ return BT_MESH_ENTER_STRING;
+ default:
+ return BT_MESH_NO_INPUT;
+ }
+}
+
+static int prov_auth(u8_t method, u8_t action, u8_t size)
+{
+ bt_mesh_output_action_t output;
+ bt_mesh_input_action_t input;
+
+ switch (method) {
+ case AUTH_METHOD_NO_OOB:
+ if (action || size) {
+ return -EINVAL;
+ }
+
+ memset(link.auth, 0, sizeof(link.auth));
+ return 0;
+ case AUTH_METHOD_STATIC:
+ if (action || size) {
+ return -EINVAL;
+ }
+
+ memcpy(link.auth + 16 - prov->static_val_len,
+ prov->static_val, prov->static_val_len);
+ memset(link.auth, 0, sizeof(link.auth) - prov->static_val_len);
+ return 0;
+
+ case AUTH_METHOD_OUTPUT:
+ output = output_action(action);
+ if (!output) {
+ return -EINVAL;
+ }
+
+ if (!(prov->output_actions & output)) {
+ return -EINVAL;
+ }
+
+ if (size > prov->output_size) {
+ return -EINVAL;
+ }
+
+ if (output == BT_MESH_DISPLAY_STRING) {
+ unsigned char str[9];
+ u8_t i;
+
+ bt_mesh_rand(str, size);
+
+ /* Normalize to '0' .. '9' & 'A' .. 'Z' */
+ for (i = 0; i < size; i++) {
+ str[i] %= 36;
+ if (str[i] < 10) {
+ str[i] += '0';
+ } else {
+ str[i] += 'A' - 10;
+ }
+ }
+ str[size] = '\0';
+
+ memcpy(link.auth, str, size);
+ memset(link.auth + size, 0, sizeof(link.auth) - size);
+
+ return prov->output_string((char *)str);
+ } else {
+ u32_t div[8] = { 10, 100, 1000, 10000, 100000,
+ 1000000, 10000000, 100000000
+ };
+ u32_t num;
+
+ bt_mesh_rand(&num, sizeof(num));
+ num %= div[size - 1];
+
+ sys_put_be32(num, &link.auth[12]);
+ memset(link.auth, 0, 12);
+
+ return prov->output_number(output, num);
+ }
+
+ case AUTH_METHOD_INPUT:
+ input = input_action(action);
+ if (!input) {
+ return -EINVAL;
+ }
+
+ if (!(prov->input_actions & input)) {
+ return -EINVAL;
+ }
+
+ if (size > prov->input_size) {
+ return -EINVAL;
+ }
+
+ if (input == BT_MESH_ENTER_STRING) {
+ atomic_set_bit(link.flags, WAIT_STRING);
+ } else {
+ atomic_set_bit(link.flags, WAIT_NUMBER);
+ }
+
+ return prov->input(input, size);
+
+ default:
+ return -EINVAL;
+ }
+}
+
+static void prov_start(const u8_t *data)
+{
+ BT_DBG("Algorithm: 0x%02x", data[0]);
+ BT_DBG("Public Key: 0x%02x", data[1]);
+ BT_DBG("Auth Method: 0x%02x", data[2]);
+ BT_DBG("Auth Action: 0x%02x", data[3]);
+ BT_DBG("Auth Size: 0x%02x", data[4]);
+
+ if (data[0] != PROV_ALG_P256) {
+ BT_ERR("Unknown algorithm 0x%02x", data[0]);
+ prov_send_fail_msg(PROV_ERR_NVAL_FMT);
+ return;
+ }
+
+ if (data[1] > 0x01) {
+ BT_ERR("Invalid public key value: 0x%02x", data[1]);
+ prov_send_fail_msg(PROV_ERR_NVAL_FMT);
+ return;
+ }
+
+ memcpy(&link.conf_inputs[12], data, 5);
+
+ link.expect = PROV_PUB_KEY;
+
+ /* If Provisioning Start PDU indicates that provisioner chooses
+ * OOB public key, then callback to the application layer to let
+ * users input public & private key pair.
+ */
+ link.oob_pk_flag = data[1] ? true : false;
+ if (link.oob_pk_flag) {
+ prov->oob_pub_key_cb();
+ }
+
+ if (prov_auth(data[2], data[3], data[4]) < 0) {
+ BT_ERR("Invalid authentication method: 0x%02x; "
+ "action: 0x%02x; size: 0x%02x", data[2], data[3],
+ data[4]);
+ prov_send_fail_msg(PROV_ERR_NVAL_FMT);
+ }
+}
+
+static void send_confirm(void)
+{
+ struct net_buf_simple *cfm = PROV_BUF(17);
+
+ BT_DBG("ConfInputs[0] %s", bt_hex(link.conf_inputs, 64));
+ BT_DBG("ConfInputs[64] %s", bt_hex(&link.conf_inputs[64], 64));
+ BT_DBG("ConfInputs[128] %s", bt_hex(&link.conf_inputs[128], 17));
+
+ if (bt_mesh_prov_conf_salt(link.conf_inputs, link.conf_salt)) {
+ BT_ERR("Unable to generate confirmation salt");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("ConfirmationSalt: %s", bt_hex(link.conf_salt, 16));
+
+ if (bt_mesh_prov_conf_key(link.dhkey, link.conf_salt, link.conf_key)) {
+ BT_ERR("Unable to generate confirmation key");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("ConfirmationKey: %s", bt_hex(link.conf_key, 16));
+
+ if (bt_mesh_rand(link.rand, 16)) {
+ BT_ERR("Unable to generate random number");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("LocalRandom: %s", bt_hex(link.rand, 16));
+
+ prov_buf_init(cfm, PROV_CONFIRM);
+
+ if (bt_mesh_prov_conf(link.conf_key, link.rand, link.auth,
+ net_buf_simple_add(cfm, 16))) {
+ BT_ERR("Unable to generate confirmation value");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (prov_send(cfm)) {
+ BT_ERR("Failed to send Provisioning Confirm");
+ close_link(PROV_ERR_RESOURCES, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ link.expect = PROV_RANDOM;
+}
+
+static void send_input_complete(void)
+{
+ struct net_buf_simple *buf = PROV_BUF(1);
+
+ prov_buf_init(buf, PROV_INPUT_COMPLETE);
+ prov_send(buf);
+}
+
+int bt_mesh_input_number(u32_t num)
+{
+ BT_DBG("%u", num);
+
+ if (!atomic_test_and_clear_bit(link.flags, WAIT_NUMBER)) {
+ return -EINVAL;
+ }
+
+ sys_put_be32(num, &link.auth[12]);
+
+ send_input_complete();
+
+ if (!atomic_test_bit(link.flags, HAVE_DHKEY)) {
+ return 0;
+ }
+
+ if (atomic_test_and_clear_bit(link.flags, SEND_CONFIRM)) {
+ send_confirm();
+ }
+
+ return 0;
+}
+
+int bt_mesh_input_string(const char *str)
+{
+ BT_DBG("%s", str);
+
+ if (!atomic_test_and_clear_bit(link.flags, WAIT_STRING)) {
+ return -EINVAL;
+ }
+
+ memcpy(link.auth, str, prov->input_size);
+
+ send_input_complete();
+
+ if (!atomic_test_bit(link.flags, HAVE_DHKEY)) {
+ return 0;
+ }
+
+ if (atomic_test_and_clear_bit(link.flags, SEND_CONFIRM)) {
+ send_confirm();
+ }
+
+ return 0;
+}
+
+static void prov_dh_key_cb(const u8_t key[32])
+{
+ BT_DBG("%p", key);
+
+ if (!key) {
+ BT_ERR("DHKey generation failed");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ sys_memcpy_swap(link.dhkey, key, 32);
+
+ BT_DBG("DHkey: %s", bt_hex(link.dhkey, 32));
+
+ atomic_set_bit(link.flags, HAVE_DHKEY);
+
+ if (atomic_test_bit(link.flags, WAIT_NUMBER) ||
+ atomic_test_bit(link.flags, WAIT_STRING)) {
+ return;
+ }
+
+ if (atomic_test_and_clear_bit(link.flags, SEND_CONFIRM)) {
+ send_confirm();
+ }
+}
+
+static void send_pub_key(void)
+{
+ struct net_buf_simple *buf = PROV_BUF(65);
+ const u8_t *key = NULL;
+
+ key = bt_mesh_pub_key_get();
+ if (!key) {
+ BT_ERR("No public key available");
+ close_link(PROV_ERR_RESOURCES, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("Local Public Key: %s", bt_hex(key, 64));
+
+ prov_buf_init(buf, PROV_PUB_KEY);
+
+ /* Swap X and Y halves independently to big-endian */
+ sys_memcpy_swap(net_buf_simple_add(buf, 32), key, 32);
+ sys_memcpy_swap(net_buf_simple_add(buf, 32), &key[32], 32);
+
+ memcpy(&link.conf_inputs[81], &buf->data[1], 64);
+
+ prov_send(buf);
+
+ /* Copy remote key in little-endian for bt_mesh_dh_key_gen().
+ * X and Y halves are swapped independently.
+ */
+ net_buf_simple_init(buf, 0);
+ sys_memcpy_swap(buf->data, &link.conf_inputs[17], 32);
+ sys_memcpy_swap(&buf->data[32], &link.conf_inputs[49], 32);
+
+ if (bt_mesh_dh_key_gen(buf->data, prov_dh_key_cb)) {
+ BT_ERR("Failed to generate DHKey");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ link.expect = PROV_CONFIRM;
+}
+
+static int bt_mesh_calc_dh_key(void)
+{
+ struct net_buf_simple *buf = NET_BUF_SIMPLE(64);
+
+ /* Copy remote key in little-endian for bt_mesh_dh_key_gen().
+ * X and Y halves are swapped independently.
+ */
+ net_buf_simple_init(buf, 0);
+ sys_memcpy_swap(buf->data, &link.conf_inputs[17], 32);
+ sys_memcpy_swap(&buf->data[32], &link.conf_inputs[49], 32);
+
+ if (bt_mesh_dh_key_gen(buf->data, prov_dh_key_cb)) {
+ BT_ERR("%s, Failed to generate DHKey", __func__);
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return -EIO;
+ }
+
+ return 0;
+}
+
+int bt_mesh_set_oob_pub_key(const u8_t pub_key_x[32], const u8_t pub_key_y[32],
+ const u8_t pri_key[32])
+{
+ if (!pub_key_x || !pub_key_y || !pri_key) {
+ BT_ERR("%s, invalid argument", __func__);
+ return -EINVAL;
+ }
+
+ /* Copy OOB public key in big-endian to Provisioning ConfirmationInputs,
+ * X and Y halves are swapped independently.
+ * And set input private key to mesh_bearer_adapt.c
+ */
+ sys_memcpy_swap(&link.conf_inputs[81], pub_key_x, 32);
+ sys_memcpy_swap(&link.conf_inputs[81] + 32, pub_key_y, 32);
+ bt_mesh_set_private_key(pri_key);
+
+ atomic_set_bit(link.flags, OOB_PUB_KEY);
+
+ /* If remote public key is not got, just return */
+ if (!atomic_test_bit(link.flags, REMOTE_PUB_KEY)) {
+ return 0;
+ }
+
+ return bt_mesh_calc_dh_key();
+}
+
+static void prov_pub_key(const u8_t *data)
+{
+ BT_DBG("Remote Public Key: %s", bt_hex(data, 64));
+
+ /* BLE Mesh BQB test case MESH/NODE/PROV/UPD/BI-13-C needs to
+ * check the public key using the following rules:
+ * (1) X > 0, Y > 0
+ * (2) X > 0, Y = 0
+ * (3) X = 0, Y = 0
+ */
+ if (!bt_mesh_check_public_key(data)) {
+ BT_ERR("%s: Invalid public key", __func__);
+ prov_send_fail_msg(PROV_ERR_UNEXP_PDU);
+ return;
+ }
+
+ memcpy(&link.conf_inputs[17], data, 64);
+ atomic_set_bit(link.flags, REMOTE_PUB_KEY);
+ /* TODO: Change by Espressif.
+ Temporary modification. Generating publish keys is not supported by the controller now */
+#if 0
+ if (!atomic_test_bit(link.flags, LOCAL_PUB_KEY)) {
+ /* Clear retransmit timer */
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ prov_clear_tx();
+#endif
+ atomic_set_bit(link.flags, REMOTE_PUB_KEY);
+ BT_WARN("Waiting for a local public key");
+ return;
+ }
+#endif /* #if 0 */
+ if (!link.oob_pk_flag) {
+ send_pub_key();
+ } else {
+ link.expect = PROV_CONFIRM;
+ }
+}
+
+#if 0
+/* Change by Espressif, currently mask this function */
+static void pub_key_ready(const u8_t *pkey)
+{
+ if (!pkey) {
+ BT_WARN("Public key not available");
+ return;
+ }
+
+ BT_DBG("Local public key ready");
+
+ atomic_set_bit(link.flags, LOCAL_PUB_KEY);
+
+ if (atomic_test_and_clear_bit(link.flags, REMOTE_PUB_KEY)) {
+ send_pub_key();
+ }
+}
+#endif
+
+static void prov_input_complete(const u8_t *data)
+{
+ BT_DBG("%s", __func__);
+}
+
+static void prov_confirm(const u8_t *data)
+{
+ BT_DBG("Remote Confirm: %s", bt_hex(data, 16));
+
+ memcpy(link.conf, data, 16);
+
+ if (!atomic_test_bit(link.flags, HAVE_DHKEY)) {
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ prov_clear_tx();
+#endif
+ atomic_set_bit(link.flags, SEND_CONFIRM);
+ /* If using OOB public key and it has already got, calculates dhkey */
+ if (link.oob_pk_flag && atomic_test_bit(link.flags, OOB_PUB_KEY)) {
+ bt_mesh_calc_dh_key();
+ }
+ } else {
+ send_confirm();
+ }
+}
+
+static void prov_random(const u8_t *data)
+{
+ struct net_buf_simple *rnd = PROV_BUF(17);
+ u8_t conf_verify[16];
+
+ BT_DBG("Remote Random: %s", bt_hex(data, 16));
+
+ if (bt_mesh_prov_conf(link.conf_key, data, link.auth, conf_verify)) {
+ BT_ERR("Unable to calculate confirmation verification");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (memcmp(conf_verify, link.conf, 16)) {
+ BT_ERR("Invalid confirmation value");
+ BT_DBG("Received: %s", bt_hex(link.conf, 16));
+ BT_DBG("Calculated: %s", bt_hex(conf_verify, 16));
+ close_link(PROV_ERR_CFM_FAILED, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ prov_buf_init(rnd, PROV_RANDOM);
+ net_buf_simple_add_mem(rnd, link.rand, 16);
+
+ if (prov_send(rnd)) {
+ BT_ERR("Failed to send Provisioning Random");
+ close_link(PROV_ERR_RESOURCES, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (bt_mesh_prov_salt(link.conf_salt, data, link.rand,
+ link.prov_salt)) {
+ BT_ERR("Failed to generate provisioning salt");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("ProvisioningSalt: %s", bt_hex(link.prov_salt, 16));
+
+ link.expect = PROV_DATA;
+}
+
+static void prov_data(const u8_t *data)
+{
+ struct net_buf_simple *msg = PROV_BUF(1);
+ u8_t session_key[16];
+ u8_t nonce[13];
+ u8_t dev_key[16];
+ u8_t pdu[25];
+ u8_t flags;
+ u32_t iv_index;
+ u16_t addr;
+ u16_t net_idx;
+ int err;
+
+ BT_DBG("%s", __func__);
+
+ err = bt_mesh_session_key(link.dhkey, link.prov_salt, session_key);
+ if (err) {
+ BT_ERR("Unable to generate session key");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("SessionKey: %s", bt_hex(session_key, 16));
+
+ err = bt_mesh_prov_nonce(link.dhkey, link.prov_salt, nonce);
+ if (err) {
+ BT_ERR("Unable to generate session nonce");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("Nonce: %s", bt_hex(nonce, 13));
+
+ err = bt_mesh_prov_decrypt(session_key, nonce, data, pdu);
+ if (err) {
+ BT_ERR("Unable to decrypt provisioning data");
+ close_link(PROV_ERR_DECRYPT, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ err = bt_mesh_dev_key(link.dhkey, link.prov_salt, dev_key);
+ if (err) {
+ BT_ERR("Unable to generate device key");
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("DevKey: %s", bt_hex(dev_key, 16));
+
+ net_idx = sys_get_be16(&pdu[16]);
+ flags = pdu[18];
+ iv_index = sys_get_be32(&pdu[19]);
+ addr = sys_get_be16(&pdu[23]);
+
+ BT_DBG("net_idx %u iv_index 0x%08x, addr 0x%04x",
+ net_idx, iv_index, addr);
+
+ prov_buf_init(msg, PROV_COMPLETE);
+ prov_send(msg);
+
+ /* Ignore any further PDUs on this link */
+ link.expect = 0;
+
+ bt_mesh_provision(pdu, net_idx, flags, iv_index, 0, addr, dev_key);
+}
+
+static void prov_complete(const u8_t *data)
+{
+ BT_DBG("%s", __func__);
+}
+
+static void prov_failed(const u8_t *data)
+{
+ BT_WARN("Error: 0x%02x", data[0]);
+}
+
+static const struct {
+ void (*func)(const u8_t *data);
+ u16_t len;
+} prov_handlers[] = {
+ { prov_invite, 1 },
+ { prov_capabilities, 11 },
+ { prov_start, 5, },
+ { prov_pub_key, 64 },
+ { prov_input_complete, 0 },
+ { prov_confirm, 16 },
+ { prov_random, 16 },
+ { prov_data, 33 },
+ { prov_complete, 0 },
+ { prov_failed, 1 },
+};
+
+static void close_link(u8_t err, u8_t reason)
+{
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (link.conn) {
+ bt_mesh_pb_gatt_close(link.conn);
+ return;
+ }
+#endif
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ if (err) {
+ prov_send_fail_msg(err);
+ }
+
+ link.rx.seg = 0;
+ bearer_ctl_send(LINK_CLOSE, &reason, sizeof(reason));
+#endif
+
+ atomic_clear_bit(link.flags, LINK_ACTIVE);
+
+ /* Disable Attention Timer if it was set before */
+ if (link.conf_inputs[0]) {
+ bt_mesh_attention(NULL, 0);
+ }
+}
+
+/* Change by Espressif, add provisioning timeout timer callback */
+static void prov_timeout(struct k_work *work)
+{
+ BT_DBG("%s", __func__);
+
+ close_link(PROV_ERR_UNEXP_ERR, CLOSE_REASON_TIMEOUT);
+}
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+static void prov_retransmit(struct k_work *work)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (!atomic_test_bit(link.flags, LINK_ACTIVE)) {
+ BT_WARN("Link not active");
+ return;
+ }
+
+ if (k_uptime_get() - link.tx.start > TRANSACTION_TIMEOUT) {
+ BT_WARN("Giving up transaction");
+ reset_link();
+ return;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(link.tx.buf); i++) {
+ struct net_buf *buf = link.tx.buf[i];
+
+ if (!buf) {
+ break;
+ }
+
+ if (BT_MESH_ADV(buf)->busy) {
+ continue;
+ }
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (i + 1 < ARRAY_SIZE(link.tx.buf) && link.tx.buf[i + 1]) {
+ bt_mesh_adv_send(buf, NULL, NULL);
+ } else {
+ bt_mesh_adv_send(buf, &buf_sent_cb, NULL);
+ }
+
+ }
+}
+
+static void link_open(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ BT_DBG("len %u", buf->len);
+
+ if (buf->len < 16) {
+ BT_ERR("Too short bearer open message (len %u)", buf->len);
+ return;
+ }
+
+ if (atomic_test_bit(link.flags, LINK_ACTIVE)) {
+ BT_WARN("Ignoring bearer open: link already active");
+ return;
+ }
+
+ if (memcmp(buf->data, prov->uuid, 16)) {
+ BT_DBG("Bearer open message not for us");
+ return;
+ }
+
+ if (prov->link_open) {
+ prov->link_open(BT_MESH_PROV_ADV);
+ }
+
+ link.id = rx->link_id;
+ atomic_set_bit(link.flags, LINK_ACTIVE);
+ net_buf_simple_init(link.rx.buf, 0);
+
+ bearer_ctl_send(LINK_ACK, NULL, 0);
+
+ link.expect = PROV_INVITE;
+}
+
+static void link_ack(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ BT_DBG("len %u", buf->len);
+}
+
+static void link_close(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ BT_DBG("len %u", buf->len);
+
+ reset_link();
+}
+
+static void gen_prov_ctl(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ BT_DBG("op 0x%02x len %u", BEARER_CTL(rx->gpc), buf->len);
+
+ switch (BEARER_CTL(rx->gpc)) {
+ case LINK_OPEN:
+ link_open(rx, buf);
+ break;
+ case LINK_ACK:
+ if (!atomic_test_bit(link.flags, LINK_ACTIVE)) {
+ return;
+ }
+
+ link_ack(rx, buf);
+ break;
+ case LINK_CLOSE:
+ if (!atomic_test_bit(link.flags, LINK_ACTIVE)) {
+ return;
+ }
+
+ link_close(rx, buf);
+ break;
+ default:
+ BT_ERR("Unknown bearer opcode: 0x%02x", BEARER_CTL(rx->gpc));
+ return;
+ }
+}
+
+static void prov_msg_recv(void)
+{
+ u8_t type = link.rx.buf->data[0];
+
+ BT_DBG("type 0x%02x len %u", type, link.rx.buf->len);
+
+ if (!bt_mesh_fcs_check(link.rx.buf, link.rx.fcs)) {
+ BT_ERR("Incorrect FCS");
+ return;
+ }
+
+ gen_prov_ack_send(link.rx.id);
+ link.rx.prev_id = link.rx.id;
+ link.rx.id = 0;
+
+ if (type != PROV_FAILED && type != link.expect) {
+ BT_WARN("Unexpected msg 0x%02x != 0x%02x", type, link.expect);
+ prov_send_fail_msg(PROV_ERR_UNEXP_PDU);
+ return;
+ }
+
+ if (type >= ARRAY_SIZE(prov_handlers)) {
+ BT_ERR("Unknown provisioning PDU type 0x%02x", type);
+ close_link(PROV_ERR_NVAL_PDU, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (1 + prov_handlers[type].len != link.rx.buf->len) {
+ BT_ERR("Invalid length %u for type 0x%02x",
+ link.rx.buf->len, type);
+ close_link(PROV_ERR_NVAL_FMT, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ /* Change by Espressif, add provisioning timeout timer operations.
+ * When received a provisioning PDU, restart the 60s timer.
+ */
+ if (atomic_test_and_clear_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link.timeout);
+ }
+ if (!atomic_test_and_set_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_submit(&link.timeout, PROVISION_TIMEOUT);
+ }
+
+ prov_handlers[type].func(&link.rx.buf->data[1]);
+}
+
+static void gen_prov_cont(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ u8_t seg = CONT_SEG_INDEX(rx->gpc);
+
+ BT_DBG("len %u, seg_index %u", buf->len, seg);
+
+ if (!link.rx.seg && link.rx.prev_id == rx->xact_id) {
+ BT_WARN("Resending ACK");
+ gen_prov_ack_send(rx->xact_id);
+ return;
+ }
+
+ if (rx->xact_id != link.rx.id) {
+ BT_WARN("Data for unknown transaction (%u != %u)",
+ rx->xact_id, link.rx.id);
+ return;
+ }
+
+ if (seg > link.rx.last_seg) {
+ BT_ERR("Invalid segment index %u", seg);
+ close_link(PROV_ERR_NVAL_FMT, CLOSE_REASON_FAILED);
+ return;
+ } else if (seg == link.rx.last_seg) {
+ u8_t expect_len;
+
+ expect_len = (link.rx.buf->len - 20 -
+ (23 * (link.rx.last_seg - 1)));
+ if (expect_len != buf->len) {
+ BT_ERR("Incorrect last seg len: %u != %u",
+ expect_len, buf->len);
+ close_link(PROV_ERR_NVAL_FMT, CLOSE_REASON_FAILED);
+ return;
+ }
+ }
+
+ if (!(link.rx.seg & BIT(seg))) {
+ BT_WARN("Ignoring already received segment");
+ return;
+ }
+
+ memcpy(XACT_SEG_DATA(seg), buf->data, buf->len);
+ XACT_SEG_RECV(seg);
+
+ if (!link.rx.seg) {
+ prov_msg_recv();
+ }
+}
+
+static void gen_prov_ack(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ BT_DBG("len %u", buf->len);
+
+ if (!link.tx.buf[0]) {
+ return;
+ }
+
+ if (rx->xact_id == link.tx.id) {
+ prov_clear_tx();
+ }
+}
+
+static void gen_prov_start(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ if (link.rx.seg) {
+ BT_WARN("Got Start while there are unreceived segments");
+ return;
+ }
+
+ if (link.rx.prev_id == rx->xact_id) {
+ BT_WARN("Resending ACK");
+ gen_prov_ack_send(rx->xact_id);
+ return;
+ }
+
+ link.rx.buf->len = net_buf_simple_pull_be16(buf);
+ link.rx.id = rx->xact_id;
+ link.rx.fcs = net_buf_simple_pull_u8(buf);
+
+ BT_DBG("len %u last_seg %u total_len %u fcs 0x%02x", buf->len,
+ START_LAST_SEG(rx->gpc), link.rx.buf->len, link.rx.fcs);
+
+ if (link.rx.buf->len < 1) {
+ BT_ERR("Ignoring zero-length provisioning PDU");
+ close_link(PROV_ERR_NVAL_FMT, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (link.rx.buf->len > link.rx.buf->size) {
+ BT_ERR("Too large provisioning PDU (%u bytes)",
+ link.rx.buf->len);
+ close_link(PROV_ERR_NVAL_FMT, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (START_LAST_SEG(rx->gpc) > 0 && link.rx.buf->len <= 20) {
+ BT_ERR("Too small total length for multi-segment PDU");
+ close_link(PROV_ERR_NVAL_FMT, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ link.rx.seg = (1 << (START_LAST_SEG(rx->gpc) + 1)) - 1;
+ link.rx.last_seg = START_LAST_SEG(rx->gpc);
+ memcpy(link.rx.buf->data, buf->data, buf->len);
+ XACT_SEG_RECV(0);
+
+ if (!link.rx.seg) {
+ prov_msg_recv();
+ }
+}
+
+static const struct {
+ void (*const func)(struct prov_rx *rx, struct net_buf_simple *buf);
+ const u8_t require_link;
+ const u8_t min_len;
+} gen_prov[] = {
+ { gen_prov_start, true, 3 },
+ { gen_prov_ack, true, 0 },
+ { gen_prov_cont, true, 0 },
+ { gen_prov_ctl, false, 0 },
+};
+
+static void gen_prov_recv(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ if (buf->len < gen_prov[GPCF(rx->gpc)].min_len) {
+ BT_ERR("Too short GPC message type %u", GPCF(rx->gpc));
+ return;
+ }
+
+ if (!atomic_test_bit(link.flags, LINK_ACTIVE) &&
+ gen_prov[GPCF(rx->gpc)].require_link) {
+ BT_DBG("Ignoring message that requires active link");
+ return;
+ }
+
+ gen_prov[GPCF(rx->gpc)].func(rx, buf);
+}
+
+void bt_mesh_pb_adv_recv(struct net_buf_simple *buf)
+{
+ struct prov_rx rx;
+
+ if (!bt_prov_active() && bt_mesh_is_provisioned()) {
+ BT_DBG("Ignoring provisioning PDU - already provisioned");
+ return;
+ }
+
+ if (buf->len < 6) {
+ BT_WARN("Too short provisioning packet (len %u)", buf->len);
+ return;
+ }
+
+ rx.link_id = net_buf_simple_pull_be32(buf);
+ rx.xact_id = net_buf_simple_pull_u8(buf);
+ rx.gpc = net_buf_simple_pull_u8(buf);
+
+ BT_DBG("link_id 0x%08x xact_id %u", rx.link_id, rx.xact_id);
+
+ if (atomic_test_bit(link.flags, LINK_ACTIVE) && link.id != rx.link_id) {
+ BT_DBG("Ignoring mesh beacon for unknown link");
+ return;
+ }
+
+ gen_prov_recv(&rx, buf);
+}
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+int bt_mesh_pb_gatt_recv(struct bt_conn *conn, struct net_buf_simple *buf)
+{
+ u8_t type;
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (link.conn != conn) {
+ BT_WARN("Data for unexpected connection");
+ return -ENOTCONN;
+ }
+
+ if (buf->len < 1) {
+ BT_WARN("Too short provisioning packet (len %u)", buf->len);
+ return -EINVAL;
+ }
+
+ type = net_buf_simple_pull_u8(buf);
+ if (type != PROV_FAILED && type != link.expect) {
+ BT_WARN("Unexpected msg 0x%02x != 0x%02x", type, link.expect);
+ prov_send_fail_msg(PROV_ERR_UNEXP_PDU);
+ return -EINVAL;
+ }
+
+ if (type >= ARRAY_SIZE(prov_handlers)) {
+ BT_ERR("Unknown provisioning PDU type 0x%02x", type);
+ return -EINVAL;
+ }
+
+ if (prov_handlers[type].len != buf->len) {
+ BT_ERR("Invalid length %u for type 0x%02x", buf->len, type);
+ return -EINVAL;
+ }
+
+ /* Change by Espressif, add provisioning timeout timer operations.
+ * When received a provisioning PDU, restart the 60s timer.
+ */
+ if (atomic_test_and_clear_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link.timeout);
+ }
+ if (!atomic_test_and_set_bit(link.flags, TIMEOUT_START)) {
+ k_delayed_work_submit(&link.timeout, PROVISION_TIMEOUT);
+ }
+
+ prov_handlers[type].func(buf->data);
+
+ return 0;
+}
+
+int bt_mesh_pb_gatt_open(struct bt_conn *conn)
+{
+ BT_DBG("conn %p", conn);
+
+ if (atomic_test_and_set_bit(link.flags, LINK_ACTIVE)) {
+ return -EBUSY;
+ }
+
+ link.conn = bt_mesh_conn_ref(conn);
+ link.expect = PROV_INVITE;
+
+ if (prov->link_open) {
+ prov->link_open(BT_MESH_PROV_GATT);
+ }
+
+ return 0;
+}
+
+int bt_mesh_pb_gatt_close(struct bt_conn *conn)
+{
+ BT_DBG("conn %p", conn);
+
+ if (link.conn != conn) {
+ BT_ERR("Not connected");
+ return -ENOTCONN;
+ }
+
+ /* Disable Attention Timer if it was set before */
+ if (link.conf_inputs[0]) {
+ bt_mesh_attention(NULL, 0);
+ }
+
+ if (prov->link_close) {
+ prov->link_close(BT_MESH_PROV_GATT);
+ }
+
+ // Should reset the atomic flag after disconneted
+ bt_mesh_conn_unref(link.conn);
+
+ /* Change by Espressif, use offsetof when clear link */
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ memset(&link, 0, offsetof(struct prov_link, tx.retransmit));
+ link.rx.prev_id = XACT_NVAL;
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ link.rx.buf = bt_mesh_proxy_get_buf();
+#else
+ net_buf_simple_init(rx_buf, 0);
+ link.rx.buf = rx_buf;
+#endif
+#else
+ memset(&link, 0, offsetof(struct prov_link, timeout));
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+ if (bt_mesh_pub_key_get()) {
+ atomic_set_bit(link.flags, LOCAL_PUB_KEY);
+ }
+
+ return 0;
+}
+#endif /* CONFIG_BT_MESH_PB_GATT */
+
+const u8_t *bt_mesh_prov_get_uuid(void)
+{
+ return prov->uuid;
+}
+
+bool bt_prov_active(void)
+{
+ return atomic_test_bit(link.flags, LINK_ACTIVE);
+}
+
+int bt_mesh_prov_init(const struct bt_mesh_prov *prov_info)
+{
+ const u8_t *key = NULL;
+
+ if (!prov_info) {
+ BT_ERR("No provisioning context provided");
+ return -EINVAL;
+ }
+
+ /* Change by Espressif. Use micro-ecc to generate public key now. */
+ key = bt_mesh_pub_key_get();
+ if (!key) {
+ BT_ERR("%s: Failed to generate public key", __func__);
+ return -EIO;
+ }
+
+ prov = prov_info;
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ k_delayed_work_init(&link.tx.retransmit, prov_retransmit);
+ link.rx.prev_id = XACT_NVAL;
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ link.rx.buf = bt_mesh_proxy_get_buf();
+#else
+ net_buf_simple_init(rx_buf, 0);
+ link.rx.buf = rx_buf;
+#endif
+
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+ if (IS_ENABLED(CONFIG_BT_DEBUG)) {
+ struct bt_uuid_128 uuid = { .uuid.type = BT_UUID_TYPE_128 };
+ memcpy(uuid.val, prov->uuid, 16);
+ BT_INFO("Device UUID: %s", bt_uuid_str(&uuid.uuid));
+ }
+
+ /* Change by Espressif, add provisioning timeout timer init */
+ k_delayed_work_init(&link.timeout, prov_timeout);
+
+ return 0;
+}
+
+void bt_mesh_prov_complete(u16_t net_idx, u16_t addr, u8_t flags, u32_t iv_index)
+{
+ if (prov->complete) {
+ prov->complete(net_idx, addr, flags, iv_index);
+ }
+}
+
+void bt_mesh_prov_reset(void)
+{
+ if (prov->reset) {
+ prov->reset();
+ }
+}
+
+#endif /* CONFIG_BT_MESH_NODE */
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/prov.h b/components/bt/ble_mesh/mesh_core/prov.h
new file mode 100644
index 0000000000..83915d8653
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/prov.h
@@ -0,0 +1,33 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef _PROV_H_
+#define _PROV_H_
+
+#include "mesh_main.h"
+#include "mesh_buf.h"
+#include "mesh_bearer_adapt.h"
+
+void bt_mesh_pb_adv_recv(struct net_buf_simple *buf);
+
+bool bt_prov_active(void);
+
+int bt_mesh_pb_gatt_open(struct bt_conn *conn);
+int bt_mesh_pb_gatt_close(struct bt_conn *conn);
+int bt_mesh_pb_gatt_recv(struct bt_conn *conn, struct net_buf_simple *buf);
+
+int bt_mesh_set_oob_pub_key(const u8_t pub_key_x[32], const u8_t pub_key_y[32],
+ const u8_t pri_key[32]);
+const u8_t *bt_mesh_prov_get_uuid(void);
+
+int bt_mesh_prov_init(const struct bt_mesh_prov *prov);
+
+void bt_mesh_prov_complete(u16_t net_idx, u16_t addr, u8_t flags, u32_t iv_index);
+void bt_mesh_prov_reset(void);
+
+#endif /* #ifndef _PROV_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_beacon.c b/components/bt/ble_mesh/mesh_core/provisioner_beacon.c
new file mode 100644
index 0000000000..24700099c5
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_beacon.c
@@ -0,0 +1,77 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "mesh_util.h"
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#include "mesh_main.h"
+
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_BEACON)
+#include "mesh_trace.h"
+
+#include "adv.h"
+#include "mesh.h"
+#include "net.h"
+#include "prov.h"
+#include "crypto.h"
+#include "beacon.h"
+#include "foundation.h"
+
+#include "provisioner_prov.h"
+
+#define BEACON_TYPE_UNPROVISIONED 0x00
+#define BEACON_TYPE_SECURE 0x01
+
+#if CONFIG_BT_MESH_PROVISIONER
+
+static void provisioner_secure_beacon_recv(struct net_buf_simple *buf)
+{
+ // TODO: Provisioner receive and handle Secure Beacon
+}
+
+void provisioner_beacon_recv(struct net_buf_simple *buf)
+{
+ u8_t type;
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (buf->len < 1) {
+ BT_ERR("Too short beacon");
+ return;
+ }
+
+ type = net_buf_simple_pull_u8(buf);
+ switch (type) {
+ case BEACON_TYPE_UNPROVISIONED:
+ BT_DBG("Unprovisioned device beacon received");
+ provisioner_unprov_beacon_recv(buf);
+ break;
+ case BEACON_TYPE_SECURE:
+ provisioner_secure_beacon_recv(buf);
+ break;
+ default:
+ BT_WARN("%s, Unknown beacon type 0x%02x", __func__, type);
+ break;
+ }
+}
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+#endif /* #if CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_beacon.h b/components/bt/ble_mesh/mesh_core/provisioner_beacon.h
new file mode 100644
index 0000000000..e1f6e7df5d
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_beacon.h
@@ -0,0 +1,22 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _PROVISIONER_BEACON_H_
+#define _PROVISIONER_BEACON_H_
+
+void provisioner_beacon_recv(struct net_buf_simple *buf);
+
+void bt_mesh_beacon_init(void);
+
+#endif /* _PROVISIONER_BEACON_H_ */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_main.c b/components/bt/ble_mesh/mesh_core/provisioner_main.c
new file mode 100644
index 0000000000..45a0d74856
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_main.c
@@ -0,0 +1,1276 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "sdkconfig.h"
+#include "osi/allocator.h"
+
+#if CONFIG_BT_MESH
+#include "mesh.h"
+#include "mesh_util.h"
+#include "mesh_main.h"
+#include "mesh_trace.h"
+#include "mesh_bearer_adapt.h"
+
+#include "crypto.h"
+#include "adv.h"
+#include "net.h"
+#include "access.h"
+
+#include "provisioner_prov.h"
+#include "provisioner_proxy.h"
+#include "provisioner_main.h"
+
+#if CONFIG_BT_MESH_PROVISIONER
+
+static const struct bt_mesh_prov *prov;
+static const struct bt_mesh_comp *comp;
+
+static struct bt_mesh_node_t *mesh_nodes[CONFIG_BT_MESH_MAX_STORED_NODES];
+static u32_t mesh_node_count;
+
+static bool prov_upper_init = false;
+
+static int provisioner_index_check(int node_index)
+{
+ struct bt_mesh_node_t *node = NULL;
+
+ BT_DBG("%s", __func__);
+
+ if (node_index < 0) {
+ BT_ERR("%s: Invalid node index %d", __func__, node_index);
+ return -EINVAL;
+ }
+
+ if (node_index >= ARRAY_SIZE(mesh_nodes)) {
+ BT_ERR("%s: Too big node index", __func__);
+ return -EINVAL;
+ }
+
+ node = mesh_nodes[node_index];
+ if (!node) {
+ BT_ERR("%s: Node is not found", __func__);
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+int provisioner_node_provision(int node_index, const u8_t uuid[16], u16_t oob_info,
+ u16_t unicast_addr, u8_t element_num, u16_t net_idx,
+ u8_t flags, u32_t iv_index, const u8_t dev_key[16])
+{
+ struct bt_mesh_node_t *node = NULL;
+
+ BT_DBG("%s", __func__);
+
+ if (mesh_node_count >= ARRAY_SIZE(mesh_nodes)) {
+ BT_ERR("%s, Node queue is full", __func__);
+ return -ENOMEM;
+ }
+
+ if (node_index >= ARRAY_SIZE(mesh_nodes) || !uuid || !dev_key) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ node = osi_calloc(sizeof(struct bt_mesh_node_t));
+ if (!node) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+
+ BT_DBG("node_index: 0x%x, unicast_addr: 0x%x, element_num: 0x%x, net_idx: 0x%x",
+ node_index, unicast_addr, element_num, net_idx);
+ BT_DBG("dev_uuid: %s", bt_hex(uuid, 16));
+ BT_DBG("dev_key: %s", bt_hex(dev_key, 16));
+
+ mesh_nodes[node_index] = node;
+
+ memcpy(node->dev_uuid, uuid, 16);
+ node->oob_info = oob_info;
+ node->unicast_addr = unicast_addr;
+ node->element_num = element_num;
+ node->net_idx = net_idx;
+ node->flags = flags;
+ node->iv_index = iv_index;
+ memcpy(node->dev_key, dev_key, 16);
+
+ mesh_node_count++;
+
+ return 0;
+}
+
+int provisioner_node_reset(int node_index)
+{
+ struct bt_mesh_node_t *node = NULL;
+ struct bt_mesh_rpl *rpl = NULL;
+ int i;
+
+ BT_DBG("%s: reset node %d", __func__, node_index);
+
+ if (!mesh_node_count) {
+ BT_ERR("%s: Node queue is empty", __func__);
+ return -ENODEV;
+ }
+
+ if (provisioner_index_check(node_index)) {
+ BT_ERR("%s: Failed to check node index", __func__);
+ return -EINVAL;
+ }
+
+ node = mesh_nodes[node_index];
+
+ /* Reset corresponding rpl when reset the node */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.rpl); i++) {
+ rpl = &bt_mesh.rpl[i];
+ if (rpl->src >= node->unicast_addr &&
+ rpl->src < node->unicast_addr + node->element_num) {
+ memset(rpl, 0, sizeof(struct bt_mesh_rpl));
+ }
+ }
+
+ osi_free(mesh_nodes[node_index]);
+ mesh_nodes[node_index] = NULL;
+
+ mesh_node_count--;
+
+ return 0;
+}
+
+int provisioner_upper_reset_all_nodes(void)
+{
+ int i, err;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ err = provisioner_node_reset(i);
+ if (err == -ENODEV) {
+ return 0;
+ }
+ }
+
+ return 0;
+}
+
+/** For Provisioner, we use the same data structure
+ * (like, struct bt_mesh_subnet, etc.) for netkey
+ * & appkey because if not we need to change a lot
+ * of APIs.
+ */
+int provisioner_upper_init(void)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ u8_t p_key[16] = {0};
+
+ BT_DBG("%s", __func__);
+
+ if (prov_upper_init) {
+ return 0;
+ }
+
+ comp = bt_mesh_comp_get();
+ if (!comp) {
+ BT_ERR("%s: NULL composition data", __func__);
+ return -EINVAL;
+ }
+
+ prov = provisioner_get_prov_info();
+ if (!prov) {
+ BT_ERR("%s: NULL provisioning context", __func__);
+ return -EINVAL;
+ }
+
+ /* If the device only acts as a Provisioner, need to initialize
+ each element's address. */
+ bt_mesh_comp_provision(prov->prov_unicast_addr);
+
+ /* Generate the primary netkey */
+ if (bt_mesh_rand(p_key, 16)) {
+ BT_ERR("%s: Failed to generate Primary NetKey", __func__);
+ return -EIO;
+ }
+
+ sub = osi_calloc(sizeof(struct bt_mesh_subnet));
+ if (!sub) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+
+ sub->kr_flag = BT_MESH_KEY_REFRESH(prov->flags);
+ if (sub->kr_flag) {
+ if (bt_mesh_net_keys_create(&sub->keys[1], p_key)) {
+ BT_ERR("%s: Failed to generate net-related keys", __func__);
+ osi_free(sub);
+ return -EIO;
+ }
+ sub->kr_phase = BT_MESH_KR_PHASE_2;
+ } else {
+ /* Currently provisioner only use keys[0] */
+ if (bt_mesh_net_keys_create(&sub->keys[0], p_key)) {
+ BT_ERR("%s: Failed to create net-related keys", __func__);
+ osi_free(sub);
+ return -EIO;
+ }
+ sub->kr_phase = BT_MESH_KR_NORMAL;
+ }
+ sub->net_idx = BT_MESH_KEY_PRIMARY;
+ sub->node_id = BT_MESH_NODE_IDENTITY_NOT_SUPPORTED;
+
+ bt_mesh.p_sub[0] = sub;
+
+ /* Dynamically added appkey & netkey will use these key_idx */
+ bt_mesh.p_app_idx_next = 0x0000;
+ bt_mesh.p_net_idx_next = 0x0001;
+
+ /* In this function, we use the values of struct bt_mesh_prov
+ which has been initialized in the application layer */
+ bt_mesh.iv_index = prov->iv_index;
+ bt_mesh.iv_update = BT_MESH_IV_UPDATE(prov->flags);
+
+ /* Set initial IV Update procedure state time-stamp */
+ bt_mesh.last_update = k_uptime_get();
+
+ prov_upper_init = true;
+
+ BT_DBG("kr_flag: %d, kr_phase: %d, net_idx: 0x%02x, node_id %d",
+ sub->kr_flag, sub->kr_phase, sub->net_idx, sub->node_id);
+ BT_DBG("netkey: %s, nid: 0x%x", bt_hex(sub->keys[0].net, 16), sub->keys[0].nid);
+ BT_DBG("enckey: %s", bt_hex(sub->keys[0].enc, 16));
+ BT_DBG("network id: %s", bt_hex(sub->keys[0].net_id, 8));
+ BT_DBG("identity: %s", bt_hex(sub->keys[0].identity, 16));
+ BT_DBG("privacy: %s", bt_hex(sub->keys[0].privacy, 16));
+ BT_DBG("beacon: %s", bt_hex(sub->keys[0].beacon, 16));
+
+ return 0;
+}
+
+/* The following APIs are for provisioner upper layers internal use */
+
+const u8_t *provisioner_net_key_get(u16_t net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (!sub || (sub->net_idx != net_idx)) {
+ continue;
+ }
+ if (sub->kr_flag) {
+ return sub->keys[1].net;
+ }
+ return sub->keys[0].net;
+ }
+
+ return NULL;
+}
+
+struct bt_mesh_subnet *provisioner_subnet_get(u16_t net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (net_idx == BT_MESH_KEY_ANY) {
+ return bt_mesh.p_sub[0];
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (!sub || (sub->net_idx != net_idx)) {
+ continue;
+ }
+ return sub;
+ }
+
+ return NULL;
+}
+
+bool provisioner_check_msg_dst_addr(u16_t dst_addr)
+{
+ struct bt_mesh_node_t *node = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (!BT_MESH_ADDR_IS_UNICAST(dst_addr)) {
+ return true;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ node = mesh_nodes[i];
+ if (node && dst_addr >= node->unicast_addr &&
+ dst_addr < node->unicast_addr + node->element_num) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+const u8_t *provisioner_get_device_key(u16_t dst_addr)
+{
+ /* Device key is only used to encrypt configuration messages.
+ * Configuration model shall only be supported by the primary
+ * element which uses the primary unicast address.
+ */
+ struct bt_mesh_node_t *node = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (!BT_MESH_ADDR_IS_UNICAST(dst_addr)) {
+ BT_ERR("%s: Not a unicast address 0x%04x", __func__, dst_addr);
+ return NULL;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ node = mesh_nodes[i];
+ if (node && node->unicast_addr == dst_addr) {
+ return node->dev_key;
+ }
+ }
+
+ return NULL;
+}
+
+struct bt_mesh_app_key *provisioner_app_key_find(u16_t app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (!key) {
+ continue;
+ }
+ if (key->net_idx != BT_MESH_KEY_UNUSED &&
+ key->app_idx == app_idx) {
+ return key;
+ }
+ }
+
+ return NULL;
+}
+
+u32_t provisioner_get_prov_node_count(void)
+{
+ return mesh_node_count;
+}
+
+/* The following APIs are for provisioner application use */
+
+#if 0
+static int bt_mesh_provisioner_set_kr_flag(u16_t net_idx, bool kr_flag)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (!sub || (sub->net_idx != net_idx)) {
+ continue;
+ }
+ sub->kr_flag = kr_flag;
+ break;
+ }
+ if (i == ARRAY_SIZE(bt_mesh.p_sub)) {
+ return -ENODEV;
+ }
+
+ /* TODO: When kr_flag is changed, provisioner may need
+ * to change the netkey of the subnet and update
+ * corresponding appkey.
+ */
+
+ return 0;
+}
+
+static void bt_mesh_provisioner_set_iv_index(u32_t iv_index)
+{
+ bt_mesh.iv_index = iv_index;
+
+ /* TODO: When iv_index is changed, provisioner may need to
+ * start iv update procedure. And the ivu_initiator
+ * & iv_update flags may also need to be set.
+ */
+}
+#endif
+
+int bt_mesh_provisioner_store_node_info(struct bt_mesh_node_t *node_info)
+{
+ struct bt_mesh_node_t *node = NULL;
+ int i;
+
+ if (!node_info) {
+ BT_ERR("%s: Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ /* Check if the device uuid already exists */
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ node = mesh_nodes[i];
+ if (node && !memcmp(node->dev_uuid, node_info->dev_uuid, 16)) {
+ BT_WARN("%s: Node info already exists", __func__);
+ return -EEXIST;
+ }
+ }
+
+ /* 0 ~ (CONFIG_BT_MESH_MAX_PROV_NODES-1) are left for self-provisioned nodes */
+ for (i = CONFIG_BT_MESH_MAX_PROV_NODES; i < ARRAY_SIZE(mesh_nodes); i++) {
+ node = mesh_nodes[i];
+ if (!node) {
+ node = osi_calloc(sizeof(struct bt_mesh_node_t));
+ if (!node) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+ memcpy(node, node_info, sizeof(struct bt_mesh_node_t));
+ mesh_nodes[i] = node;
+ mesh_node_count++;
+ return 0;
+ }
+ }
+
+ BT_ERR("%s: Node info is full", __func__);
+ return -ENOMEM;
+}
+
+int bt_mesh_provisioner_get_all_node_unicast_addr(struct net_buf_simple *buf)
+{
+ struct bt_mesh_node_t *node = NULL;
+ int i;
+
+ if (!buf) {
+ BT_ERR("%s: Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ node = mesh_nodes[i];
+ if (!node || !BT_MESH_ADDR_IS_UNICAST(node->unicast_addr)) {
+ continue;
+ }
+ net_buf_simple_add_le16(buf, node->unicast_addr);
+ }
+
+ return 0;
+}
+
+int bt_mesh_provisioner_set_node_name(int node_index, const char *name)
+{
+ size_t length, name_len;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (!name) {
+ BT_ERR("%s: Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ if (provisioner_index_check(node_index)) {
+ BT_ERR("%s: Failed to check node index", __func__);
+ return -EINVAL;
+ }
+
+ BT_DBG("name len is %d, name is %s", strlen(name), name);
+
+ length = (strlen(name) <= MESH_NAME_SIZE) ? strlen(name) : MESH_NAME_SIZE;
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ if (!mesh_nodes[i] || !mesh_nodes[i]->node_name) {
+ continue;
+ }
+ name_len = strlen(mesh_nodes[i]->node_name);
+ if (length != name_len) {
+ continue;
+ }
+ if (!strncmp(mesh_nodes[i]->node_name, name, length)) {
+ BT_WARN("%s: Name %s already exists", __func__, name);
+ return -EEXIST;
+ }
+ }
+
+ strncpy(mesh_nodes[node_index]->node_name, name, length);
+
+ return 0;
+}
+
+const char *bt_mesh_provisioner_get_node_name(int node_index)
+{
+ BT_DBG("%s", __func__);
+
+ if (provisioner_index_check(node_index)) {
+ BT_ERR("%s: Failed to check node index", __func__);
+ return NULL;
+ }
+
+ return mesh_nodes[node_index]->node_name;
+}
+
+int bt_mesh_provisioner_get_node_index(const char *name)
+{
+ size_t length, name_len;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (!name) {
+ return -EINVAL;
+ }
+
+ length = (strlen(name) <= MESH_NAME_SIZE) ? strlen(name) : MESH_NAME_SIZE;
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ if (!mesh_nodes[i] || !mesh_nodes[i]->node_name) {
+ continue;
+ }
+ name_len = strlen(mesh_nodes[i]->node_name);
+ if (length != name_len) {
+ continue;
+ }
+ if (!strncmp(mesh_nodes[i]->node_name, name, length)) {
+ return i;
+ }
+ }
+
+ return -ENODEV;
+}
+
+struct bt_mesh_node_t *bt_mesh_provisioner_get_node_info(u16_t unicast_addr)
+{
+ struct bt_mesh_node_t *node = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (!BT_MESH_ADDR_IS_UNICAST(unicast_addr)) {
+ BT_ERR("%s: Not a unicast address 0x%04x", __func__, unicast_addr);
+ return NULL;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ node = mesh_nodes[i];
+ if (!node) {
+ continue;
+ }
+ if (unicast_addr >= node->unicast_addr &&
+ unicast_addr < (node->unicast_addr + node->element_num)) {
+ return node;
+ }
+ }
+
+ return NULL;
+}
+
+u32_t bt_mesh_provisioner_get_net_key_count(void)
+{
+ return ARRAY_SIZE(bt_mesh.p_sub);
+}
+
+u32_t bt_mesh_provisioner_get_app_key_count(void)
+{
+ return ARRAY_SIZE(bt_mesh.p_app_keys);
+}
+
+static int provisioner_check_app_key(const u8_t app_key[16], u16_t *app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+ int i;
+
+ if (!app_key) {
+ return 0;
+ }
+
+ /* Check if app_key is already existed */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (key && (!memcmp(key->keys[0].val, app_key, 16) ||
+ !memcmp(key->keys[1].val, app_key, 16))) {
+ *app_idx = key->app_idx;
+ return -EEXIST;
+ }
+ }
+
+ return 0;
+}
+
+static int provisioner_check_app_idx(u16_t app_idx, bool exist)
+{
+ struct bt_mesh_app_key *key = NULL;
+ int i;
+
+ if (exist) {
+ /* Check if app_idx is already existed */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (key && (key->app_idx == app_idx)) {
+ return -EEXIST;
+ }
+ }
+ return 0;
+ }
+
+ /* Check if app_idx is not existed */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (key && (key->app_idx == app_idx)) {
+ return 0;
+ }
+ }
+
+ return -ENODEV;
+}
+
+static int provisioner_check_app_key_full(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ if (!bt_mesh.p_app_keys[i]) {
+ return i;
+ }
+ }
+
+ return -ENOMEM;
+}
+
+static int provisioner_check_net_key(const u8_t net_key[16], u16_t *net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ int i;
+
+ if (!net_key) {
+ return 0;
+ }
+
+ /* Check if net_key is already existed */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (sub && (!memcmp(sub->keys[0].net, net_key, 16) ||
+ !memcmp(sub->keys[1].net, net_key, 16))) {
+ *net_idx = sub->net_idx;
+ return -EEXIST;
+ }
+ }
+
+ return 0;
+}
+
+static int provisioner_check_net_idx(u16_t net_idx, bool exist)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ int i;
+
+ if (exist) {
+ /* Check if net_idx is already existed */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (sub && (sub->net_idx == net_idx)) {
+ return -EEXIST;
+ }
+ }
+ return 0;
+ }
+
+ /* Check if net_idx is not existed */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (sub && (sub->net_idx == net_idx)) {
+ return 0;
+ }
+ }
+
+ return -ENODEV;
+}
+
+static int provisioner_check_net_key_full(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ if (!bt_mesh.p_sub[i]) {
+ return i;
+ }
+ }
+
+ return -ENOMEM;
+}
+
+int bt_mesh_provisioner_local_app_key_add(const u8_t app_key[16], u16_t net_idx, u16_t *app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+ struct bt_mesh_app_keys *keys = NULL;
+ u8_t p_key[16] = {0};
+ int add = -1;
+
+ if (bt_mesh.p_app_idx_next >= 0x1000) {
+ BT_ERR("%s: No AppKey Index available", __func__);
+ return -EIO;
+ }
+
+ if (!app_idx || (*app_idx != 0xFFFF && *app_idx >= 0x1000)) {
+ BT_ERR("%s: Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ /* Check if the same application key already exists */
+ if (provisioner_check_app_key(app_key, app_idx)) {
+ BT_WARN("%s: AppKey already exists, AppKey Index updated", __func__);
+ return 0;
+ }
+
+ /* Check if the net_idx exists */
+ if (provisioner_check_net_idx(net_idx, false)) {
+ BT_ERR("%s: NetKey Index does not exist", __func__);
+ return -ENODEV;
+ }
+
+ /* Check if the same app_idx already exists */
+ if (provisioner_check_app_idx(*app_idx, true)) {
+ BT_ERR("%s: AppKey Index already exists", __func__);
+ return -EEXIST;
+ }
+
+ add = provisioner_check_app_key_full();
+ if (add < 0) {
+ BT_ERR("%s: AppKey queue is full", __func__);
+ return -ENOMEM;
+ }
+
+ if (!app_key) {
+ if (bt_mesh_rand(p_key, 16)) {
+ BT_ERR("%s: Failed to generate AppKey", __func__);
+ return -EIO;
+ }
+ } else {
+ memcpy(p_key, app_key, 16);
+ }
+
+ key = osi_calloc(sizeof(struct bt_mesh_app_key));
+ if (!key) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+
+ keys = &key->keys[0];
+ if (bt_mesh_app_id(p_key, &keys->id)) {
+ BT_ERR("%s: Failed to generate AID", __func__);
+ osi_free(key);
+ return -EIO;
+ }
+
+ memcpy(keys->val, p_key, 16);
+ key->net_idx = net_idx;
+ if (*app_idx != 0xFFFF) {
+ key->app_idx = *app_idx;
+ } else {
+ key->app_idx = bt_mesh.p_app_idx_next;
+ while (1) {
+ if (provisioner_check_app_idx(key->app_idx, true)) {
+ key->app_idx = (++bt_mesh.p_app_idx_next);
+ if (key->app_idx >= 0x1000) {
+ BT_ERR("%s: No AppKey Index available", __func__);
+ osi_free(key);
+ return -EIO;
+ }
+ } else {
+ break;
+ }
+ }
+ *app_idx = key->app_idx;
+ }
+ key->updated = false;
+
+ bt_mesh.p_app_keys[add] = key;
+
+ return 0;
+}
+
+const u8_t *bt_mesh_provisioner_local_app_key_get(u16_t net_idx, u16_t app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (provisioner_check_net_idx(net_idx, false)) {
+ BT_ERR("%s: NetKey Index does not exist", __func__);
+ return NULL;
+ }
+
+ if (provisioner_check_app_idx(app_idx, false)) {
+ BT_ERR("%s: AppKey Index does not exist", __func__);
+ return NULL;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (key && key->net_idx == net_idx &&
+ key->app_idx == app_idx) {
+ if (key->updated) {
+ return key->keys[1].val;
+ }
+ return key->keys[0].val;
+ }
+ }
+
+ return NULL;
+}
+
+int bt_mesh_provisioner_local_app_key_delete(u16_t net_idx, u16_t app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (provisioner_check_net_idx(net_idx, false)) {
+ BT_ERR("%s: NetKey Index does not exist", __func__);
+ return -ENODEV;
+ }
+
+ if (provisioner_check_app_idx(app_idx, false)) {
+ BT_ERR("%s: AppKey Index does not exist", __func__);
+ return -ENODEV;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (key && key->net_idx == net_idx &&
+ key->app_idx == app_idx) {
+ osi_free(bt_mesh.p_app_keys[i]);
+ bt_mesh.p_app_keys[i] = NULL;
+ return 0;
+ }
+ }
+
+ /* Shall never reach here */
+ return -ENODEV;
+}
+
+int bt_mesh_provisioner_local_net_key_add(const u8_t net_key[16], u16_t *net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ u8_t p_key[16] = {0};
+ int add = -1;
+
+ if (bt_mesh.p_net_idx_next >= 0x1000) {
+ BT_ERR("%s: No NetKey Index available", __func__);
+ return -EIO;
+ }
+
+ if (!net_idx || (*net_idx != 0xFFFF && *net_idx >= 0x1000)) {
+ BT_ERR("%s: Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ /* Check if the same network key already exists */
+ if (provisioner_check_net_key(net_key, net_idx)) {
+ BT_WARN("%s: NetKey already exists, NetKey Index updated", __func__);
+ return 0;
+ }
+
+ /* Check if the same net_idx already exists */
+ if (provisioner_check_net_idx(*net_idx, true)) {
+ BT_ERR("%s: NetKey Index already exists", __func__);
+ return -EEXIST;
+ }
+
+ add = provisioner_check_net_key_full();
+ if (add < 0) {
+ BT_ERR("%s: NetKey queue is full", __func__);
+ return -ENOMEM;
+ }
+
+ if (!net_key) {
+ if (bt_mesh_rand(p_key, 16)) {
+ BT_ERR("%s: Failed to generate NetKey", __func__);
+ return -EIO;
+ }
+ } else {
+ memcpy(p_key, net_key, 16);
+ }
+
+ sub = osi_calloc(sizeof(struct bt_mesh_subnet));
+ if (!sub) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+
+ if (bt_mesh_net_keys_create(&sub->keys[0], p_key)) {
+ BT_ERR("%s: Failed to generate NID", __func__);
+ osi_free(sub);
+ return -EIO;
+ }
+
+ if (*net_idx != 0xFFFF) {
+ sub->net_idx = *net_idx;
+ } else {
+ sub->net_idx = bt_mesh.p_net_idx_next;
+ while (1) {
+ if (provisioner_check_net_idx(sub->net_idx, true)) {
+ sub->net_idx = (++bt_mesh.p_net_idx_next);
+ if (sub->net_idx >= 0x1000) {
+ BT_ERR("%s: No NetKey Index available", __func__);
+ osi_free(sub);
+ return -EIO;
+ }
+ } else {
+ break;
+ }
+ }
+ *net_idx = sub->net_idx;
+ }
+ sub->kr_phase = BT_MESH_KR_NORMAL;
+ sub->kr_flag = false;
+ sub->node_id = BT_MESH_NODE_IDENTITY_NOT_SUPPORTED;
+
+ bt_mesh.p_sub[add] = sub;
+
+ return 0;
+}
+
+const u8_t *bt_mesh_provisioner_local_net_key_get(u16_t net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (provisioner_check_net_idx(net_idx, false)) {
+ BT_ERR("%s: NetKey Index does not exist", __func__);
+ return NULL;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (sub && sub->net_idx == net_idx) {
+ if (sub->kr_flag) {
+ return sub->keys[1].net;
+ }
+ return sub->keys[0].net;
+ }
+ }
+
+ return NULL;
+}
+
+int bt_mesh_provisioner_local_net_key_delete(u16_t net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (provisioner_check_net_idx(net_idx, false)) {
+ BT_ERR("%s: NetKey Index does not exist", __func__);
+ return -ENODEV;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (sub && sub->net_idx == net_idx) {
+ osi_free(bt_mesh.p_sub[i]);
+ bt_mesh.p_sub[i] = NULL;
+ return 0;
+ }
+ }
+
+ /* Shall never reach here */
+ return -ENODEV;
+}
+
+int bt_mesh_provisioner_get_own_unicast_addr(u16_t *addr, u8_t *elem_num)
+{
+ if (!addr || !elem_num || !prov || !comp) {
+ BT_ERR("%s: Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ *addr = prov->prov_unicast_addr;
+ *elem_num = comp->elem_count;
+
+ return 0;
+}
+
+int bt_mesh_provisioner_bind_local_model_app_idx(u16_t elem_addr, u16_t mod_id,
+ u16_t cid, u16_t app_idx)
+{
+ struct bt_mesh_elem *elem = NULL;
+ struct bt_mesh_model *model = NULL;
+ int i;
+
+ if (!comp) {
+ BT_ERR("%s: NULL composition data", __func__);
+ return -EINVAL;
+ }
+
+ for (i = 0; i < comp->elem_count; i++) {
+ elem = &comp->elem[i];
+ if (elem->addr == elem_addr) {
+ break;
+ }
+ }
+ if (i == comp->elem_count) {
+ BT_ERR("%s: No element is found", __func__);
+ return -ENODEV;
+ }
+
+ if (cid == 0xFFFF) {
+ model = bt_mesh_model_find(elem, mod_id);
+ } else {
+ model = bt_mesh_model_find_vnd(elem, cid, mod_id);
+ }
+ if (!model) {
+ BT_ERR("%s: No model is found", __func__);
+ return -ENODEV;
+ }
+
+ if (provisioner_check_app_idx(app_idx, false)) {
+ BT_ERR("%s: AppKey Index does not exist", __func__);
+ return -ENODEV;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(model->keys); i++) {
+ if (model->keys[i] == app_idx) {
+ BT_WARN("%s: AppKey Index is already binded with model", __func__);
+ return 0;
+ }
+ }
+
+ for (i = 0; i < ARRAY_SIZE(model->keys); i++) {
+ if (model->keys[i] == BT_MESH_KEY_UNUSED) {
+ model->keys[i] = app_idx;
+ return 0;
+ }
+ }
+
+ BT_ERR("%s: Model AppKey queue is full", __func__);
+ return -ENOMEM;
+}
+
+int bt_mesh_provisioner_bind_local_app_net_idx(u16_t net_idx, u16_t app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ if (provisioner_check_net_idx(net_idx, false)) {
+ BT_ERR("%s: NetKey Index does not exist", __func__);
+ return -ENODEV;
+ }
+
+ if (provisioner_check_app_idx(app_idx, false)) {
+ BT_ERR("%s: AppKey Index does not exist", __func__);
+ return -ENODEV;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (!key || (key->app_idx != app_idx)) {
+ continue;
+ }
+ key->net_idx = net_idx;
+ return 0;
+ }
+
+ return -ENODEV;
+}
+
+int bt_mesh_provisioner_print_local_element_info(void)
+{
+ struct bt_mesh_elem *elem = NULL;
+ struct bt_mesh_model *model = NULL;
+ int i, j;
+
+ if (!comp) {
+ BT_ERR("%s: NULL composition data", __func__);
+ return -EINVAL;
+ }
+
+ BT_WARN("************************************************");
+ BT_WARN("* cid: 0x%04x pid: 0x%04x vid: 0x%04x *", comp->cid, comp->pid, comp->vid);
+ BT_WARN("* Element Number: 0x%02x *", comp->elem_count);
+ for (i = 0; i < comp->elem_count; i++) {
+ elem = &comp->elem[i];
+ BT_WARN("* Element %d: 0x%04x *", i, elem->addr);
+ BT_WARN("* Loc: 0x%04x NumS: 0x%02x NumV: 0x%02x *", elem->loc, elem->model_count, elem->vnd_model_count);
+ for (j = 0; j < elem->model_count; j++) {
+ model = &elem->models[j];
+ BT_WARN("* sig_model %d: id - 0x%04x *", j, model->id);
+ }
+ for (j = 0; j < elem->vnd_model_count; j++) {
+ model = &elem->vnd_models[j];
+ BT_WARN("* vnd_model %d: id - 0x%04x, cid - 0x%04x *", j, model->vnd.id, model->vnd.company);
+ }
+ }
+ BT_WARN("************************************************");
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+/* The following APIs are for fast provisioning */
+
+#if CONFIG_BT_MESH_FAST_PROV
+
+const u8_t *get_fast_prov_device_key(u16_t addr)
+{
+ struct bt_mesh_node_t *node = NULL;
+
+ BT_DBG("%s", __func__);
+
+ if (!BT_MESH_ADDR_IS_UNICAST(addr)) {
+ BT_ERR("%s: Not a unicast address 0x%04x", __func__, addr);
+ return NULL;
+ }
+
+ if (addr == bt_mesh_primary_addr()) {
+ return bt_mesh.dev_key;
+ }
+
+ for (int i = 0; i < ARRAY_SIZE(mesh_nodes); i++) {
+ node = mesh_nodes[i];
+ if (node && node->unicast_addr == addr) {
+ return node->dev_key;
+ }
+ }
+
+ return NULL;
+}
+
+struct bt_mesh_subnet *get_fast_prov_subnet(u16_t net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+
+ BT_DBG("%s", __func__);
+
+ for (int i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ sub = &bt_mesh.sub[i];
+ if (sub->net_idx == net_idx) {
+ return sub;
+ }
+ }
+
+ for (int i = 0; i < ARRAY_SIZE(bt_mesh.p_sub); i++) {
+ sub = bt_mesh.p_sub[i];
+ if (sub && sub->net_idx == net_idx) {
+ return sub;
+ }
+ }
+
+ return NULL;
+}
+
+struct bt_mesh_app_key *get_fast_prov_app_key(u16_t net_idx, u16_t app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+
+ BT_DBG("%s", __func__);
+
+ for (int i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ key = &bt_mesh.app_keys[i];
+ if (key->net_idx == net_idx && key->app_idx == app_idx) {
+ return key;
+ }
+ }
+
+ for (int i = 0; i < ARRAY_SIZE(bt_mesh.p_app_keys); i++) {
+ key = bt_mesh.p_app_keys[i];
+ if (key && key->net_idx == net_idx && key->app_idx == app_idx) {
+ return key;
+ }
+ }
+
+ return NULL;
+}
+
+u8_t bt_mesh_set_fast_prov_net_idx(u16_t net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+ struct bt_mesh_subnet_keys *key = NULL;
+
+ sub = get_fast_prov_subnet(net_idx);
+ if (sub) {
+ key = BT_MESH_KEY_REFRESH(sub->kr_flag) ? &sub->keys[1] : &sub->keys[0];
+ return provisioner_set_fast_prov_net_idx(key->net, net_idx);
+ }
+
+ /* If net_idx is not found, set net_idx to fast_prov first,
+ * and wait for primary provisioner to add net_key */
+ return provisioner_set_fast_prov_net_idx(NULL, net_idx);
+}
+
+u8_t bt_mesh_add_fast_prov_net_key(const u8_t net_key[16])
+{
+ const u8_t *keys = NULL;
+ u16_t net_idx;
+ int err;
+
+ net_idx = provisioner_get_fast_prov_net_idx();
+ bt_mesh.p_net_idx_next = net_idx;
+
+ err = bt_mesh_provisioner_local_net_key_add(net_key, &net_idx);
+ if (err) {
+ return 0x01; /* status: add net_key fail */
+ };
+
+ keys = bt_mesh_provisioner_local_net_key_get(net_idx);
+ if (!keys) {
+ return 0x01; /* status: add net_key fail */
+ }
+
+ return provisioner_set_fast_prov_net_idx(keys, net_idx);
+}
+
+const u8_t *bt_mesh_get_fast_prov_net_key(u16_t net_idx)
+{
+ struct bt_mesh_subnet *sub = NULL;
+
+ sub = get_fast_prov_subnet(net_idx);
+ if (!sub) {
+ BT_ERR("%s: Failed to get subnet", __func__);
+ return NULL;
+ }
+
+ return (sub->kr_flag ? sub->keys[1].net : sub->keys[0].net);
+}
+
+const u8_t *bt_mesh_get_fast_prov_app_key(u16_t net_idx, u16_t app_idx)
+{
+ struct bt_mesh_app_key *key = NULL;
+
+ key = get_fast_prov_app_key(net_idx, app_idx);
+ if (!key) {
+ BT_ERR("%s: Failed to get AppKey", __func__);
+ return NULL;
+ }
+
+ return (key->updated ? key->keys[1].val : key->keys[0].val);
+}
+
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+#endif /* CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_main.h b/components/bt/ble_mesh/mesh_core/provisioner_main.h
new file mode 100644
index 0000000000..d7fb7faeb6
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_main.h
@@ -0,0 +1,130 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _PROVISIONER_MAIN_H_
+#define _PROVISIONER_MAIN_H_
+
+#include "mesh_util.h"
+#include "mesh_kernel.h"
+#include "mesh_access.h"
+#include "net.h"
+
+#define MESH_NAME_SIZE 31
+
+#if CONFIG_BT_MESH_PROVISIONER
+
+/* Each node information stored by provisioner */
+struct bt_mesh_node_t {
+ char node_name[MESH_NAME_SIZE]; /* Node name */
+ u8_t dev_uuid[16]; /* Device UUID pointer, stored in provisioner_prov.c */
+ u16_t oob_info; /* Node OOB information */
+ u16_t unicast_addr; /* Node unicast address */
+ u8_t element_num; /* Node element number */
+ u16_t net_idx; /* Node provision net_idx */
+ u8_t flags; /* Node key refresh flag and iv update flag */
+ u32_t iv_index; /* Node IV Index */
+ u8_t dev_key[16]; /* Node device key */
+} __packed;
+
+/* The following APIs are for key init, node provision & node reset. */
+
+int provisioner_node_provision(int node_index, const u8_t uuid[16], u16_t oob_info,
+ u16_t unicast_addr, u8_t element_num, u16_t net_idx,
+ u8_t flags, u32_t iv_index, const u8_t dev_key[16]);
+
+int provisioner_node_reset(int node_index);
+
+int provisioner_upper_reset_all_nodes(void);
+
+int provisioner_upper_init(void);
+
+/* The following APIs are for provisioner upper layers internal usage. */
+
+const u8_t *provisioner_net_key_get(u16_t net_idx);
+
+struct bt_mesh_subnet *provisioner_subnet_get(u16_t net_idx);
+
+bool provisioner_check_msg_dst_addr(u16_t dst_addr);
+
+const u8_t *provisioner_get_device_key(u16_t dst_addr);
+
+struct bt_mesh_app_key *provisioner_app_key_find(u16_t app_idx);
+
+u32_t provisioner_get_prov_node_count(void);
+
+/* The following APIs are for provisioner application use. */
+
+int bt_mesh_provisioner_store_node_info(struct bt_mesh_node_t *node_info);
+
+int bt_mesh_provisioner_get_all_node_unicast_addr(struct net_buf_simple *buf);
+
+int bt_mesh_provisioner_set_node_name(int node_index, const char *name);
+
+const char *bt_mesh_provisioner_get_node_name(int node_index);
+
+int bt_mesh_provisioner_get_node_index(const char *name);
+
+struct bt_mesh_node_t *bt_mesh_provisioner_get_node_info(u16_t unicast_addr);
+
+u32_t bt_mesh_provisioner_get_net_key_count(void);
+
+u32_t bt_mesh_provisioner_get_app_key_count(void);
+
+int bt_mesh_provisioner_local_app_key_add(const u8_t app_key[16], u16_t net_idx, u16_t *app_idx);
+
+const u8_t *bt_mesh_provisioner_local_app_key_get(u16_t net_idx, u16_t app_idx);
+
+int bt_mesh_provisioner_local_app_key_delete(u16_t net_idx, u16_t app_idx);
+
+int bt_mesh_provisioner_local_net_key_add(const u8_t net_key[16], u16_t *net_idx);
+
+const u8_t *bt_mesh_provisioner_local_net_key_get(u16_t net_idx);
+
+int bt_mesh_provisioner_local_net_key_delete(u16_t net_idx);
+
+int bt_mesh_provisioner_get_own_unicast_addr(u16_t *addr, u8_t *elem_num);
+
+/* Provisioner bind local client model with proper appkey index */
+int bt_mesh_provisioner_bind_local_model_app_idx(u16_t elem_addr, u16_t mod_id,
+ u16_t cid, u16_t app_idx);
+
+/* This API can be used to change the net_idx binded with the app_idx. */
+int bt_mesh_provisioner_bind_local_app_net_idx(u16_t net_idx, u16_t app_idx);
+
+/* Provisioner print own element information */
+int bt_mesh_provisioner_print_local_element_info(void);
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+/* The following APIs are for fast provisioning */
+
+#if CONFIG_BT_MESH_FAST_PROV
+
+const u8_t *get_fast_prov_device_key(u16_t dst_addr);
+
+struct bt_mesh_subnet *get_fast_prov_subnet(u16_t net_idx);
+
+struct bt_mesh_app_key *get_fast_prov_app_key(u16_t net_idx, u16_t app_idx);
+
+u8_t bt_mesh_set_fast_prov_net_idx(u16_t net_idx);
+
+u8_t bt_mesh_add_fast_prov_net_key(const u8_t net_key[16]);
+
+const u8_t *bt_mesh_get_fast_prov_net_key(u16_t net_idx);
+
+const u8_t *bt_mesh_get_fast_prov_app_key(u16_t net_idx, u16_t app_idx);
+
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+#endif /* _PROVISIONER_MAIN_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_prov.c b/components/bt/ble_mesh/mesh_core/provisioner_prov.c
new file mode 100644
index 0000000000..aecbec3500
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_prov.c
@@ -0,0 +1,3207 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "stack/bt_types.h"
+#include "sdkconfig.h"
+#include "osi/allocator.h"
+
+#if CONFIG_BT_MESH
+#include "mesh.h"
+#include "mesh_main.h"
+#include "mesh_trace.h"
+
+#include "crypto.h"
+#include "adv.h"
+#include "provisioner_prov.h"
+#include "provisioner_proxy.h"
+#include "provisioner_main.h"
+
+#if CONFIG_BT_MESH_PROVISIONER
+
+#define CONFIG_BT_MAX_CONN CONFIG_BT_ACL_CONNECTIONS
+
+/* Service data length has minus 1 type length & 2 uuid length*/
+#define BT_MESH_PROV_SRV_DATA_LEN 0x12
+#define BT_MESH_PROXY_SRV_DATA_LEN1 0x09
+#define BT_MESH_PROXY_SRV_DATA_LEN2 0x11
+
+/* 3 transmissions, 20ms interval */
+#define PROV_XMIT_COUNT 2
+#define PROV_XMIT_INT 20
+
+#define AUTH_METHOD_NO_OOB 0x00
+#define AUTH_METHOD_STATIC 0x01
+#define AUTH_METHOD_OUTPUT 0x02
+#define AUTH_METHOD_INPUT 0x03
+
+#define OUTPUT_OOB_BLINK 0x00
+#define OUTPUT_OOB_BEEP 0x01
+#define OUTPUT_OOB_VIBRATE 0x02
+#define OUTPUT_OOB_NUMBER 0x03
+#define OUTPUT_OOB_STRING 0x04
+
+#define INPUT_OOB_PUSH 0x00
+#define INPUT_OOB_TWIST 0x01
+#define INPUT_OOB_NUMBER 0x02
+#define INPUT_OOB_STRING 0x03
+
+#define PROV_ERR_NONE 0x00
+#define PROV_ERR_NVAL_PDU 0x01
+#define PROV_ERR_NVAL_FMT 0x02
+#define PROV_ERR_UNEXP_PDU 0x03
+#define PROV_ERR_CFM_FAILED 0x04
+#define PROV_ERR_RESOURCES 0x05
+#define PROV_ERR_DECRYPT 0x06
+#define PROV_ERR_UNEXP_ERR 0x07
+#define PROV_ERR_ADDR 0x08
+
+#define PROV_INVITE 0x00
+#define PROV_CAPABILITIES 0x01
+#define PROV_START 0x02
+#define PROV_PUB_KEY 0x03
+#define PROV_INPUT_COMPLETE 0x04
+#define PROV_CONFIRM 0x05
+#define PROV_RANDOM 0x06
+#define PROV_DATA 0x07
+#define PROV_COMPLETE 0x08
+#define PROV_FAILED 0x09
+
+#define PROV_ALG_P256 0x00
+
+#define GPCF(gpc) (gpc & 0x03)
+#define GPC_START(last_seg) (((last_seg) << 2) | 0x00)
+#define GPC_ACK 0x01
+#define GPC_CONT(seg_id) (((seg_id) << 2) | 0x02)
+#define GPC_CTL(op) (((op) << 2) | 0x03)
+
+#define START_PAYLOAD_MAX 20
+#define CONT_PAYLOAD_MAX 23
+
+#define START_LAST_SEG(gpc) (gpc >> 2)
+#define CONT_SEG_INDEX(gpc) (gpc >> 2)
+
+#define BEARER_CTL(gpc) (gpc >> 2)
+#define LINK_OPEN 0x00
+#define LINK_ACK 0x01
+#define LINK_CLOSE 0x02
+
+#define CLOSE_REASON_SUCCESS 0x00
+#define CLOSE_REASON_TIMEOUT 0x01
+#define CLOSE_REASON_FAILED 0x02
+
+#define PROV_AUTH_VAL_SIZE 0x10
+#define PROV_CONF_SALT_SIZE 0x10
+#define PROV_CONF_KEY_SIZE 0x10
+#define PROV_DH_KEY_SIZE 0x20
+#define PROV_CONFIRM_SIZE 0x10
+#define PROV_PROV_SALT_SIZE 0x10
+#define PROV_CONF_INPUTS_SIZE 0x91
+
+static inline int prov_get_pb_index(void);
+
+#define XACT_SEG_DATA(_seg) (&link[prov_get_pb_index()].rx.buf->data[20 + ((_seg - 1) * 23)])
+#define XACT_SEG_RECV(_seg) (link[prov_get_pb_index()].rx.seg &= ~(1 << (_seg)))
+
+#define XACT_NVAL 0xff
+
+enum {
+ REMOTE_PUB_KEY, /* Remote key has been received */
+ LOCAL_PUB_KEY, /* Local public key is available */
+ LINK_ACTIVE, /* Link has been opened */
+ WAIT_GEN_DHKEY, /* Waiting for remote public key to generate DHKey */
+ HAVE_DHKEY, /* DHKey has been calcualted */
+ SEND_CONFIRM, /* Waiting to send Confirm value */
+ WAIT_NUMBER, /* Waiting for number input from user */
+ WAIT_STRING, /* Waiting for string input from user */
+ TIMEOUT_START, /* Provision timeout timer has started */
+ NUM_FLAGS,
+};
+
+/** Provisioner link structure allocation
+ * |--------------------------------------------------------|
+ * | Link(PB-ADV) | Link(PB-GATT) |
+ * |--------------------------------------------------------|
+ * |<----------------------Total Link---------------------->|
+ */
+struct prov_link {
+ ATOMIC_DEFINE(flags, NUM_FLAGS);
+ u8_t uuid[16]; /* check if device is being provisioned*/
+ u16_t oob_info; /* oob info of this device */
+ u8_t element_num; /* element num of device */
+ u8_t ki_flags; /* Key refresh flag and iv update flag */
+ u32_t iv_index; /* IV Index */
+ u8_t auth_method; /* choosed authentication method */
+ u8_t auth_action; /* choosed authentication action */
+ u8_t auth_size; /* choosed authentication size */
+ u16_t unicast_addr; /* unicast address assigned for device */
+ bt_addr_le_t addr; /* Device address */
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ bool connecting; /* start connecting with device */
+ struct bt_conn *conn; /* GATT connection */
+#endif
+ u8_t expect; /* Next expected PDU */
+
+ u8_t *dhkey; /* Calculated DHKey */
+ u8_t *auth; /* Authentication Value */
+
+ u8_t *conf_salt; /* ConfirmationSalt */
+ u8_t *conf_key; /* ConfirmationKey */
+ u8_t *conf_inputs; /* ConfirmationInputs */
+
+ u8_t *rand; /* Local Random */
+ u8_t *conf; /* Remote Confirmation */
+
+ u8_t *prov_salt; /* Provisioning Salt */
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ bool linking; /* Linking is being establishing */
+ u16_t link_close; /* Link close been sent flag */
+ u32_t link_id; /* Link ID */
+ u8_t pending_ack; /* Decide which transaction id ack is pending */
+ u8_t expect_ack_for; /* Transaction ACK expected for provisioning pdu */
+
+ struct {
+ u8_t trans_id; /* Transaction ID */
+ u8_t prev_id; /* Previous Transaction ID */
+ u8_t seg; /* Bit-field of unreceived segments */
+ u8_t last_seg; /* Last segment (to check length) */
+ u8_t fcs; /* Expected FCS value */
+ u8_t adv_buf_id; /* index of buf allocated in adv_buf_data */
+ struct net_buf_simple *buf;
+ } rx;
+
+ struct {
+ /* Start timestamp of the transaction */
+ s64_t start;
+
+ /* Transaction id*/
+ u8_t trans_id;
+
+ /* Pending outgoing buffer(s) */
+ struct net_buf *buf[3];
+
+ /* Retransmit timer */
+ struct k_delayed_work retransmit;
+ } tx;
+#endif
+ /** Provision timeout timer. Spec P259 says: The provisioning protocol
+ * shall have a minimum timeout of 60 seconds that is reset each time
+ * a provisioning protocol PDU is sent or received.
+ */
+ struct k_delayed_work timeout;
+};
+
+struct prov_rx {
+ u32_t link_id;
+ u8_t xact_id;
+ u8_t gpc;
+};
+
+#define BT_MESH_ALREADY_PROV_NUM (CONFIG_BT_MESH_MAX_PROV_NODES + 10)
+
+struct prov_ctx_t {
+ /* If provisioning random have been generated, set BIT0 to 1 */
+ u8_t rand_gen_done;
+
+ /* Provisioner random */
+ u8_t random[16];
+
+ /* Number of provisioned devices */
+ u16_t node_count;
+
+ /* Current number of PB-ADV provisioned devices simultaneously */
+ u8_t pba_count;
+
+ /* Current number of PB-GATT provisioned devices simultaneously */
+ u8_t pbg_count;
+
+ /* Current index of device being provisioned using PB-GATT or PB-ADV */
+ int pb_index;
+
+ /* Current unicast address going to assigned */
+ u16_t current_addr;
+
+ /* Current net_idx going to be used in provisioning data */
+ u16_t curr_net_idx;
+
+ /* Current flags going to be used in provisioning data */
+ u16_t curr_flags;
+
+ /* Current iv_index going to be used in provisioning data */
+ u16_t curr_iv_index;
+
+ /* Offset of the device uuid to be matched, based on zero */
+ u8_t match_offset;
+
+ /* Length of the device uuid to be matched (start from the match_offset) */
+ u8_t match_length;
+
+ /* Value of the device uuid to be matched */
+ u8_t *match_value;
+
+ /* Indicate when received uuid_match adv_pkts, can provision it at once */
+ bool prov_after_match;
+
+ /** This structure is used to store the information of the device which
+ * provisioner has successfully sent provisioning data to. In this
+ * structure, we don't care if the device is currently in the mesh
+ * network, or has been removed, or failed to send provisioning
+ * complete pdu after receiving the provisioning data pdu.
+ */
+ struct already_prov_info {
+ u8_t uuid[16]; /* device uuid */
+ u8_t element_num; /* element number of the deleted node */
+ u16_t unicast_addr; /* Primary unicast address of the deleted node */
+ } already_prov[BT_MESH_ALREADY_PROV_NUM];
+};
+
+struct prov_node_info {
+ bool provisioned; /* device provisioned flag */
+ bt_addr_le_t addr; /* device address */
+ u8_t uuid[16]; /* node uuid */
+ u16_t oob_info; /* oob info contained in adv pkt */
+ u8_t element_num; /* element contained in this node */
+ u16_t unicast_addr; /* primary unicast address of this node */
+ u16_t net_idx; /* Netkey index got during provisioning */
+ u8_t flags; /* Key refresh flag and iv update flag */
+ u32_t iv_index; /* IV Index */
+};
+
+struct unprov_dev_queue {
+ bt_addr_le_t addr;
+ u8_t uuid[16];
+ u16_t oob_info;
+ u8_t bearer;
+ u8_t flags;
+} __packed unprov_dev[CONFIG_BT_MESH_UNPROV_DEV_ADD] = {
+ [0 ... (CONFIG_BT_MESH_UNPROV_DEV_ADD - 1)] = {
+ .addr.type = 0xff,
+ .bearer = 0,
+ .flags = false,
+ },
+};
+
+static unprov_adv_pkt_cb_t notify_unprov_adv_pkt_cb;
+
+#define RETRANSMIT_TIMEOUT K_MSEC(500)
+#define BUF_TIMEOUT K_MSEC(400)
+#if defined(CONFIG_BT_MESH_FAST_PROV)
+#define TRANSACTION_TIMEOUT K_SECONDS(3)
+#define PROVISION_TIMEOUT K_SECONDS(6)
+#else
+#define TRANSACTION_TIMEOUT K_SECONDS(30)
+#define PROVISION_TIMEOUT K_SECONDS(60)
+#endif /* CONFIG_BT_MESH_FAST_PROV */
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+#define PROV_BUF_HEADROOM 5
+#else
+#define PROV_BUF_HEADROOM 0
+#endif
+
+#define PROV_BUF(len) NET_BUF_SIMPLE(PROV_BUF_HEADROOM + len)
+
+/* Number of devices can be provisioned at the same time using PB-GATT + PB-ADV */
+#define BT_MESH_PROV_SAME_TIME (CONFIG_BT_MESH_PBA_SAME_TIME + CONFIG_BT_MESH_PBG_SAME_TIME)
+
+static struct prov_link link[BT_MESH_PROV_SAME_TIME];
+
+static const struct bt_mesh_prov *prov;
+
+static struct prov_ctx_t prov_ctx;
+
+static struct prov_node_info prov_nodes[CONFIG_BT_MESH_MAX_PROV_NODES];
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+static void send_link_open(void);
+#endif
+
+static void prov_gen_dh_key(int i);
+
+static void send_pub_key(u8_t oob);
+
+static void close_link(int i, u8_t reason);
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+#define ADV_BUF_SIZE 65
+static struct adv_buf_t {
+ struct net_buf_simple buf;
+ u8_t adv_buf_data[ADV_BUF_SIZE];
+} adv_buf[CONFIG_BT_MESH_PBA_SAME_TIME];
+#endif
+
+#define PROV_FREE_MEM(id, member) \
+{ \
+ if (link[id].member) { \
+ osi_free(link[id].member); \
+ } \
+}
+
+/* Fast provisioning uses this structure for provisioning data */
+struct bt_mesh_fast_prov {
+ u16_t net_idx;
+ const u8_t *net_key;
+ u8_t flags;
+ u32_t iv_index;
+ u16_t unicast_addr_min;
+ u16_t unicast_addr_max;
+};
+static struct bt_mesh_fast_prov fast_prov;
+static bool fast_prov_flag;
+#define FAST_PROV_FLAG_GET() fast_prov_flag
+
+static inline int prov_get_pb_index(void)
+{
+ return prov_ctx.pb_index;
+}
+
+static void prov_set_pb_index(int i)
+{
+ prov_ctx.pb_index = i;
+}
+
+void provisioner_pbg_count_dec(void)
+{
+ if (prov_ctx.pbg_count) {
+ prov_ctx.pbg_count--;
+ }
+}
+
+void provisioner_pbg_count_inc(void)
+{
+ prov_ctx.pbg_count++;
+}
+
+void provisioner_clear_link_conn_info(u8_t addr[6])
+{
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (!addr) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return;
+ }
+
+ BT_DBG("%s, Clear device %s info", __func__, bt_hex(addr, BD_ADDR_LEN));
+
+ for (int i = CONFIG_BT_MESH_PBA_SAME_TIME; i < BT_MESH_PROV_SAME_TIME; i++) {
+ if (!memcmp(link[i].addr.a.val, addr, BD_ADDR_LEN)) {
+ link[i].connecting = false;
+ link[i].conn = NULL;
+ link[i].oob_info = 0x0;
+ memset(link[i].uuid, 0, 16);
+ memset(&link[i].addr, 0, sizeof(bt_addr_le_t));
+ atomic_test_and_clear_bit(link[i].flags, LINK_ACTIVE);
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+ return;
+ }
+ }
+
+ BT_WARN("%s, Address %s is not found", __func__, bt_hex(addr, BD_ADDR_LEN));
+ return;
+#endif
+}
+
+const struct bt_mesh_prov *provisioner_get_prov_info(void)
+{
+ return prov;
+}
+
+int provisioner_prov_reset_all_nodes(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(prov_nodes); i++) {
+ if (prov_nodes[i].provisioned) {
+ memset(&prov_nodes[i], 0, sizeof(struct prov_node_info));
+ }
+ }
+
+ prov_ctx.node_count = 0;
+
+ return 0;
+}
+
+static int provisioner_dev_find(const bt_addr_le_t *addr, const u8_t uuid[16], int *index)
+{
+ bool uuid_match = false;
+ bool addr_match = false;
+ u8_t zero[16] = {0};
+ int i = 0, j = 0, comp = 0;
+
+ if (addr) {
+ comp = memcmp(addr->a.val, zero, BD_ADDR_LEN);
+ }
+
+ if ((!uuid && (!addr || (comp == 0) || (addr->type > BLE_ADDR_RANDOM))) || !index) {
+ return -EINVAL;
+ }
+
+ /** Note: user may add a device into two unprov_dev array elements,
+ * one with device address, address type and another only
+ * with device UUID. We need to take this into consideration.
+ */
+ if (uuid && memcmp(uuid, zero, 16)) {
+ for (i = 0; i < ARRAY_SIZE(unprov_dev); i++) {
+ if (!memcmp(unprov_dev[i].uuid, uuid, 16)) {
+ uuid_match = true;
+ break;
+ }
+ }
+ }
+
+ if (addr && comp && (addr->type <= BLE_ADDR_RANDOM)) {
+ for (j = 0; j < ARRAY_SIZE(unprov_dev); j++) {
+ if (!memcmp(unprov_dev[j].addr.a.val, addr->a.val, BD_ADDR_LEN) &&
+ unprov_dev[j].addr.type == addr->type) {
+ addr_match = true;
+ break;
+ }
+ }
+ }
+
+ if (!uuid_match && !addr_match) {
+ BT_DBG("%s, Device does not exist in queue", __func__);
+ return -ENODEV;
+ }
+
+ if (uuid_match && addr_match && (i != j)) {
+ /** In this situation, copy address & type into device
+ * uuid array element, reset another element, rm_flag
+ * will be decided by uuid element.
+ */
+ unprov_dev[i].addr.type = unprov_dev[j].addr.type;
+ memcpy(unprov_dev[i].addr.a.val, unprov_dev[j].addr.a.val, BD_ADDR_LEN);
+ unprov_dev[i].bearer |= unprov_dev[j].bearer;
+ memset(&unprov_dev[j], 0x0, sizeof(struct unprov_dev_queue));
+ }
+
+ *index = uuid_match ? i : j;
+ return 0;
+}
+
+static int provisioner_dev_uuid_match(const u8_t uuid[16])
+{
+ if (!uuid) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ if (prov_ctx.match_length && prov_ctx.match_value) {
+ if (memcmp(uuid + prov_ctx.match_offset,
+ prov_ctx.match_value, prov_ctx.match_length)) {
+ return -EAGAIN;
+ }
+ }
+
+ return 0;
+}
+
+static int provisioner_check_device_uuid(const u8_t uuid[16])
+{
+ int i;
+
+ if (!uuid) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ /* Check if this device is currently being provisioned.
+ * According to Zephyr's device code, if we connect with
+ * one device and start to provision it, we may still can
+ * receive the connectable prov adv pkt from this device.
+ * Here we check both PB-GATT and PB-ADV link status.
+ */
+ for (i = 0; i < BT_MESH_PROV_SAME_TIME; i++) {
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (link[i].connecting || atomic_test_bit(link[i].flags, LINK_ACTIVE)) {
+#else
+ if (link[i].linking || atomic_test_bit(link[i].flags, LINK_ACTIVE)) {
+#endif
+ if (!memcmp(link[i].uuid, uuid, 16)) {
+ BT_DBG("%s, Device is being provisioned", __func__);
+ return -EALREADY;
+ }
+ }
+ }
+
+ /* Check if the device uuid matches configured value */
+ if (provisioner_dev_uuid_match(uuid)) {
+ BT_DBG("%s, Device uuid failed to match", __func__);
+ return -EIO;
+ }
+
+ /* Check if the device has already been provisioned */
+ for (i = 0; i < ARRAY_SIZE(prov_nodes); i++) {
+ if (prov_nodes[i].provisioned) {
+ if (!memcmp(prov_nodes[i].uuid, uuid, 16)) {
+ BT_WARN("Provisioned before, start to provision again");
+ provisioner_node_reset(i);
+ memset(&prov_nodes[i], 0, sizeof(struct prov_node_info));
+ if (prov_ctx.node_count) {
+ prov_ctx.node_count--;
+ }
+ return 0;
+ }
+ }
+ }
+
+ /* Check if the prov_nodes queue is full */
+ if (prov_ctx.node_count == ARRAY_SIZE(prov_nodes)) {
+ BT_WARN("Current provisioned devices reach max limit");
+ return -ENOMEM;
+ }
+
+ return 0;
+}
+
+static int provisioner_start_prov_device(bt_mesh_prov_bearer_t bearer, const u8_t uuid[16],
+ const bt_addr_le_t *addr, u16_t oob_info)
+{
+ u8_t zero[6] = {0};
+ int addr_cmp, i;
+
+ if ((bearer != BT_MESH_PROV_ADV && bearer != BT_MESH_PROV_GATT) || !uuid || !addr) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ addr_cmp = memcmp(addr->a.val, zero, BD_ADDR_LEN);
+
+ if (bearer == BT_MESH_PROV_ADV) {
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ for (i = 0; i < CONFIG_BT_MESH_PBA_SAME_TIME; i++) {
+ if (!atomic_test_bit(link[i].flags, LINK_ACTIVE) && !link[i].linking) {
+ memcpy(link[i].uuid, uuid, 16);
+ link[i].oob_info = oob_info;
+ if (addr_cmp && (addr->type <= BLE_ADDR_RANDOM)) {
+ link[i].addr.type = addr->type;
+ memcpy(link[i].addr.a.val, addr->a.val, BD_ADDR_LEN);
+ }
+ prov_set_pb_index(i);
+ send_link_open();
+ /* If Provisioner sets LINK_ACTIVE flag once Link Open is sent, here
+ * we may not need to use linking flag (like PB-GATT connecting) to
+ * prevent the stored device info (UUID, oob_info) being replaced by
+ * other received unprovisioned device beacons.
+ * But if Provisioner sets LINK_ACTIVE flag after Link ACK is received,
+ * we need to use linking flag to prevent device info being replaced.
+ * Currently we set LINK_ACTIVE flag after sending Link Open.
+ */
+ link[i].linking = true;
+ if (prov->prov_link_open) {
+ prov->prov_link_open(BT_MESH_PROV_ADV);
+ }
+ return 0;
+ }
+ }
+#endif
+ } else {
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ for (i = CONFIG_BT_MESH_PBA_SAME_TIME; i < BT_MESH_PROV_SAME_TIME; i++) {
+ if (!atomic_test_bit(link[i].flags, LINK_ACTIVE) && !link[i].connecting) {
+ memcpy(link[i].uuid, uuid, 16);
+ link[i].oob_info = oob_info;
+ if (addr_cmp && (addr->type <= BLE_ADDR_RANDOM)) {
+ link[i].addr.type = addr->type;
+ memcpy(link[i].addr.a.val, addr->a.val, BD_ADDR_LEN);
+ }
+ if (bt_mesh_gattc_conn_create(&link[i].addr, BT_UUID_MESH_PROV_VAL)) {
+ memset(link[i].uuid, 0, 16);
+ link[i].oob_info = 0x0;
+ memset(&link[i].addr, 0, sizeof(bt_addr_le_t));
+ return -EIO;
+ }
+ /* If creating connection successfully, set connecting flag to 1 */
+ link[i].connecting = true;
+ return 0;
+ }
+ }
+#endif
+ }
+
+ BT_ERR("%s, no link is available", __func__);
+ return -ENOMEM;
+}
+
+int bt_mesh_provisioner_add_unprov_dev(struct bt_mesh_unprov_dev_add *add_dev, u8_t flags)
+{
+ bt_addr_le_t add_addr = {0};
+ u8_t zero[16] = {0};
+ int addr_cmp = 0, uuid_cmp = 0;
+ int i, err = 0;
+
+ if (!add_dev) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ addr_cmp = memcmp(add_dev->addr, zero, BD_ADDR_LEN);
+ uuid_cmp = memcmp(add_dev->uuid, zero, 16);
+
+ if (add_dev->bearer == 0x0 || ((uuid_cmp == 0) &&
+ ((addr_cmp == 0) || add_dev->addr_type > BLE_ADDR_RANDOM))) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ if ((add_dev->bearer & BT_MESH_PROV_ADV) && (add_dev->bearer & BT_MESH_PROV_GATT) &&
+ (flags & START_PROV_NOW)) {
+ BT_ERR("%s, Can not start PB-ADV & PB-GATT simultaneouly", __func__);
+ return -EINVAL;
+ }
+
+ if ((uuid_cmp == 0) && (flags & START_PROV_NOW)) {
+ BT_ERR("%s, Can not start provisioning with zero uuid", __func__);
+ return -EINVAL;
+ }
+
+ if ((add_dev->bearer & BT_MESH_PROV_GATT) && (flags & START_PROV_NOW) &&
+ ((addr_cmp == 0) || add_dev->addr_type > BLE_ADDR_RANDOM)) {
+ BT_ERR("%s, Invalid device address for PB-GATT", __func__);
+ return -EINVAL;
+ }
+
+ if (add_dev->bearer & BT_MESH_PROV_GATT) {
+#if !CONFIG_BT_MESH_PB_GATT
+ BT_ERR("%s, Not support PB-GATT", __func__);
+ return -EINVAL;
+#endif
+ }
+
+ if (add_dev->bearer & BT_MESH_PROV_ADV) {
+#if !CONFIG_BT_MESH_PB_ADV
+ BT_ERR("%s, Not support PB-ADV", __func__);
+ return -EINVAL;
+#endif
+ }
+
+ add_addr.type = add_dev->addr_type;
+ memcpy(add_addr.a.val, add_dev->addr, BD_ADDR_LEN);
+
+ err = provisioner_dev_find(&add_addr, add_dev->uuid, &i);
+ if (err == -EINVAL) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return err;
+ } else if (err == 0) {
+ if (!(add_dev->bearer & unprov_dev[i].bearer)) {
+ BT_WARN("%s, Add device with only bearer updated", __func__);
+ unprov_dev[i].bearer |= add_dev->bearer;
+ } else {
+ BT_WARN("%s, Device already exists", __func__);
+ }
+ goto start;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(unprov_dev); i++) {
+ if (unprov_dev[i].bearer) {
+ continue;
+ }
+ if (addr_cmp && (add_dev->addr_type <= BLE_ADDR_RANDOM)) {
+ unprov_dev[i].addr.type = add_dev->addr_type;
+ memcpy(unprov_dev[i].addr.a.val, add_dev->addr, BD_ADDR_LEN);
+ }
+ if (uuid_cmp) {
+ memcpy(unprov_dev[i].uuid, add_dev->uuid, 16);
+ }
+ unprov_dev[i].bearer = add_dev->bearer & BIT_MASK(2);
+ unprov_dev[i].flags = flags & BIT_MASK(3);
+ goto start;
+ }
+
+ /* If queue is full, find flushable device and replace it */
+ for (i = 0; i < ARRAY_SIZE(unprov_dev); i++) {
+ if (unprov_dev[i].flags & FLUSHABLE_DEV) {
+ memset(&unprov_dev[i], 0, sizeof(struct unprov_dev_queue));
+ if (addr_cmp && (add_dev->addr_type <= BLE_ADDR_RANDOM)) {
+ unprov_dev[i].addr.type = add_dev->addr_type;
+ memcpy(unprov_dev[i].addr.a.val, add_dev->addr, BD_ADDR_LEN);
+ }
+ if (uuid_cmp) {
+ memcpy(unprov_dev[i].uuid, add_dev->uuid, 16);
+ }
+ unprov_dev[i].bearer = add_dev->bearer & BIT_MASK(2);
+ unprov_dev[i].flags = flags & BIT_MASK(3);
+ goto start;
+ }
+ }
+
+ BT_ERR("%s, Unprovisioned device queue is full", __func__);
+ return -ENOMEM;
+
+start:
+ if (!(flags & START_PROV_NOW)) {
+ return 0;
+ }
+
+ /* Check if current provisioned node count + active link reach max limit */
+ if (prov_ctx.node_count + prov_ctx.pba_count + \
+ prov_ctx.pbg_count >= ARRAY_SIZE(prov_nodes)) {
+ BT_WARN("%s, Node count + active link count reach max limit", __func__);
+ return -EIO;
+ }
+
+ if ((err = provisioner_check_device_uuid(add_dev->uuid))) {
+ return err;
+ }
+
+ if (add_dev->bearer & BT_MESH_PROV_ADV) {
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ if (prov_ctx.pba_count == CONFIG_BT_MESH_PBA_SAME_TIME) {
+ BT_WARN("%s, Current PB-ADV links reach max limit", __func__);
+ return -EIO;
+ }
+ if ((err = provisioner_start_prov_device(BT_MESH_PROV_ADV,
+ add_dev->uuid, &add_addr, add_dev->oob_info))) {
+ return err;
+ }
+#endif
+ } else if (add_dev->bearer & BT_MESH_PROV_GATT) {
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (prov_ctx.pbg_count == CONFIG_BT_MESH_PBG_SAME_TIME) {
+ BT_WARN("%s, Current PB-GATT links reach max limit", __func__);
+ return -EIO;
+ }
+ if ((err = provisioner_start_prov_device(BT_MESH_PROV_GATT,
+ add_dev->uuid, &add_addr, add_dev->oob_info))) {
+ return err;
+ }
+#endif
+ }
+
+ return 0;
+}
+
+int bt_mesh_provisioner_delete_device(struct bt_mesh_device_delete *del_dev)
+{
+ /**
+ * Three Situations:
+ * 1. device is not being/been provisioned, just remove from device queue.
+ * 2. device is being provisioned, need to close link & remove from device queue.
+ * 3. device is been provisioned, need to send config_node_reset and may need to
+ * remove from device queue. config _node_reset can be added in function
+ * provisioner_node_reset() in provisioner_main.c.
+ */
+ bt_addr_le_t del_addr = {0};
+ u8_t zero[16] = {0};
+ int addr_cmp = 0, uuid_cmp = 0;
+ bool addr_match = false;
+ bool uuid_match = false;
+ int i, err = 0;
+
+ if (!del_dev) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ addr_cmp = memcmp(del_dev->addr, zero, BD_ADDR_LEN);
+ uuid_cmp = memcmp(del_dev->uuid, zero, 16);
+
+ if ((uuid_cmp == 0) && ((addr_cmp == 0) || del_dev->addr_type > BLE_ADDR_RANDOM)) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ del_addr.type = del_dev->addr_type;
+ memcpy(del_addr.a.val, del_dev->addr, BD_ADDR_LEN);
+
+ /* First: find if the device is in the device queue */
+ err = provisioner_dev_find(&del_addr, del_dev->uuid, &i);
+ if (err) {
+ BT_DBG("%s, Device is not in the queue", __func__);
+ } else {
+ memset(&unprov_dev[i], 0x0, sizeof(struct unprov_dev_queue));
+ }
+
+ /* Second: find if the device is being provisioned */
+ for (i = 0; i < ARRAY_SIZE(link); i++) {
+ if (addr_cmp && (del_dev->addr_type <= BLE_ADDR_RANDOM)) {
+ if (!memcmp(link[i].addr.a.val, del_dev->addr, BD_ADDR_LEN) &&
+ link[i].addr.type == del_dev->addr_type) {
+ addr_match = true;
+ }
+ }
+ if (uuid_cmp) {
+ if (!memcmp(link[i].uuid, del_dev->uuid, 16)) {
+ uuid_match = true;
+ }
+ }
+ if (addr_match || uuid_match) {
+ close_link(i, CLOSE_REASON_FAILED);
+ break;
+ }
+ }
+
+ /* Third: find if the device is been provisioned */
+ for (i = 0; i < ARRAY_SIZE(prov_nodes); i++) {
+ if (addr_cmp && (del_dev->addr_type <= BLE_ADDR_RANDOM)) {
+ if (!memcmp(prov_nodes[i].addr.a.val, del_dev->addr, BD_ADDR_LEN) &&
+ prov_nodes[i].addr.type == del_dev->addr_type) {
+ addr_match = true;
+ }
+ }
+ if (uuid_cmp) {
+ if (!memcmp(prov_nodes[i].uuid, del_dev->uuid, 16)) {
+ uuid_match = true;
+ }
+ }
+ if (addr_match || uuid_match) {
+ memset(&prov_nodes[i], 0, sizeof(struct prov_node_info));
+ provisioner_node_reset(i);
+ if (prov_ctx.node_count) {
+ prov_ctx.node_count--;
+ }
+ break;
+ }
+ }
+
+ return 0;
+}
+
+int bt_mesh_provisioner_set_dev_uuid_match(u8_t offset, u8_t length,
+ const u8_t *match, bool prov_flag)
+{
+ if (length && (!match || (offset + length > 16))) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ if (length && !prov_ctx.match_value) {
+ prov_ctx.match_value = osi_calloc(16);
+ if (!prov_ctx.match_value) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ return -ENOMEM;
+ }
+ }
+
+ prov_ctx.match_offset = offset;
+ prov_ctx.match_length = length;
+ if (length) {
+ memcpy(prov_ctx.match_value, match, length);
+ }
+ prov_ctx.prov_after_match = prov_flag;
+
+ return 0;
+}
+
+int bt_mesh_prov_adv_pkt_cb_register(unprov_adv_pkt_cb_t cb)
+{
+ if (!cb) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ notify_unprov_adv_pkt_cb = cb;
+ return 0;
+}
+
+int bt_mesh_provisioner_set_prov_data_info(struct bt_mesh_prov_data_info *info)
+{
+ const u8_t *key = NULL;
+
+ if (!info || info->flag == 0) {
+ return -EINVAL;
+ }
+
+ if (info->flag & NET_IDX_FLAG) {
+ key = provisioner_net_key_get(info->net_idx);
+ if (!key) {
+ BT_ERR("%s, Failed to get NetKey", __func__);
+ return -EINVAL;
+ }
+ prov_ctx.curr_net_idx = info->net_idx;
+ } else if (info->flag & FLAGS_FLAG) {
+ prov_ctx.curr_flags = info->flags;
+ } else if (info->flag & IV_INDEX_FLAG) {
+ prov_ctx.curr_iv_index = info->iv_index;
+ }
+
+ return 0;
+}
+
+/* The following APIs are for fast provisioning */
+
+void provisioner_set_fast_prov_flag(bool flag)
+{
+ fast_prov_flag = flag;
+}
+
+u8_t provisioner_set_fast_prov_net_idx(const u8_t *net_key, u16_t net_idx)
+{
+ fast_prov.net_idx = net_idx;
+ fast_prov.net_key = net_key;
+
+ if (!net_key) {
+ BT_WARN("%s, Wait for NetKey for fast provisioning", __func__);
+ return 0x01; /*status: wait for net_key */
+ }
+
+ return 0x0; /* status: success */
+}
+
+u16_t provisioner_get_fast_prov_net_idx(void)
+{
+ return fast_prov.net_idx;
+}
+
+u8_t bt_mesh_set_fast_prov_unicast_addr_range(u16_t min, u16_t max)
+{
+ if (!BT_MESH_ADDR_IS_UNICAST(min) || !BT_MESH_ADDR_IS_UNICAST(max)) {
+ BT_ERR("%s, Not a unicast address", __func__);
+ return 0x01; /* status: not a unicast address */
+ }
+
+ if (min > max) {
+ BT_ERR("%s, Min bigger than max", __func__);
+ return 0x02; /* status: min is bigger than max */
+ }
+
+ if (min <= fast_prov.unicast_addr_max) {
+ BT_ERR("%s, Address overlap", __func__);
+ return 0x03; /* status: address overlaps with current value */
+ }
+
+ fast_prov.unicast_addr_min = min;
+ fast_prov.unicast_addr_max = max;
+
+ prov_ctx.current_addr = fast_prov.unicast_addr_min;
+
+ return 0x0; /* status: success */
+}
+
+void bt_mesh_set_fast_prov_flags_iv_index(u8_t flags, u32_t iv_index)
+{
+ /* BIT0: Key Refreash flag, BIT1: IV Update flag */
+ fast_prov.flags = flags & BIT_MASK(2);
+ fast_prov.iv_index = iv_index;
+}
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+static struct net_buf_simple *bt_mesh_pba_get_buf(int id)
+{
+ struct net_buf_simple *buf = &(adv_buf[id].buf);
+
+ net_buf_simple_init(buf, 0);
+
+ return buf;
+}
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+static void prov_memory_free(int i)
+{
+ PROV_FREE_MEM(i, dhkey);
+ PROV_FREE_MEM(i, auth);
+ PROV_FREE_MEM(i, conf);
+ PROV_FREE_MEM(i, conf_salt);
+ PROV_FREE_MEM(i, conf_key);
+ PROV_FREE_MEM(i, conf_inputs);
+ PROV_FREE_MEM(i, prov_salt);
+}
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+static void buf_sent(int err, void *user_data)
+{
+ int i = (int)user_data;
+
+ if (!link[i].tx.buf[0]) {
+ return;
+ }
+
+ k_delayed_work_submit(&link[i].tx.retransmit, RETRANSMIT_TIMEOUT);
+}
+
+static struct bt_mesh_send_cb buf_sent_cb = {
+ .end = buf_sent,
+};
+
+static void free_segments(int id)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(link[id].tx.buf); i++) {
+ struct net_buf *buf = link[id].tx.buf[i];
+
+ if (!buf) {
+ break;
+ }
+
+ link[id].tx.buf[i] = NULL;
+ /* Mark as canceled */
+ BT_MESH_ADV(buf)->busy = 0;
+ /** Change by Espressif. Add this to avoid buf->ref is 2 which will
+ * cause lack of buf.
+ */
+ if (buf->ref > 1) {
+ buf->ref = 1;
+ }
+ net_buf_unref(buf);
+ }
+}
+
+static void prov_clear_tx(int i)
+{
+ BT_DBG("%s", __func__);
+
+ k_delayed_work_cancel(&link[i].tx.retransmit);
+
+ free_segments(i);
+}
+
+static void reset_link(int i, u8_t reason)
+{
+ prov_clear_tx(i);
+
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ if (prov->prov_link_close) {
+ prov->prov_link_close(BT_MESH_PROV_ADV, reason);
+ }
+
+ prov_memory_free(i);
+
+ /* Clear everything except the retransmit delayed work config */
+ memset(&link[i], 0, offsetof(struct prov_link, tx.retransmit));
+
+ link[i].pending_ack = XACT_NVAL;
+ link[i].rx.prev_id = XACT_NVAL;
+
+ if (bt_mesh_pub_key_get()) {
+ atomic_set_bit(link[i].flags, LOCAL_PUB_KEY);
+ }
+
+ link[i].rx.buf = bt_mesh_pba_get_buf(i);
+
+ if (prov_ctx.pba_count) {
+ prov_ctx.pba_count--;
+ }
+}
+
+static struct net_buf *adv_buf_create(void)
+{
+ struct net_buf *buf;
+
+ buf = bt_mesh_adv_create(BT_MESH_ADV_PROV, PROV_XMIT_COUNT,
+ PROV_XMIT_INT, BUF_TIMEOUT);
+ if (!buf) {
+ BT_ERR("Out of provisioning buffers");
+ return NULL;
+ }
+
+ return buf;
+}
+
+static void ack_complete(u16_t duration, int err, void *user_data)
+{
+ int i = (int)user_data;
+
+ BT_DBG("xact %u complete", (u8_t)link[i].pending_ack);
+
+ link[i].pending_ack = XACT_NVAL;
+}
+
+static void gen_prov_ack_send(u8_t xact_id)
+{
+ static const struct bt_mesh_send_cb cb = {
+ .start = ack_complete,
+ };
+ const struct bt_mesh_send_cb *complete;
+ struct net_buf *buf;
+ int i = prov_get_pb_index();
+
+ BT_DBG("xact_id %u", xact_id);
+
+ if (link[i].pending_ack == xact_id) {
+ BT_DBG("Not sending duplicate ack");
+ return;
+ }
+
+ buf = adv_buf_create();
+ if (!buf) {
+ return;
+ }
+
+ if (link[i].pending_ack == XACT_NVAL) {
+ link[i].pending_ack = xact_id;
+ complete = &cb;
+ } else {
+ complete = NULL;
+ }
+
+ net_buf_add_be32(buf, link[i].link_id);
+ net_buf_add_u8(buf, xact_id);
+ net_buf_add_u8(buf, GPC_ACK);
+
+ bt_mesh_adv_send(buf, complete, (void *)i);
+ net_buf_unref(buf);
+}
+
+static void send_reliable(int id)
+{
+ link[id].tx.start = k_uptime_get();
+
+ for (int i = 0; i < ARRAY_SIZE(link[id].tx.buf); i++) {
+ struct net_buf *buf = link[id].tx.buf[i];
+
+ if (!buf) {
+ break;
+ }
+
+ if (i + 1 < ARRAY_SIZE(link[id].tx.buf) && link[id].tx.buf[i + 1]) {
+ bt_mesh_adv_send(buf, NULL, NULL);
+ } else {
+ bt_mesh_adv_send(buf, &buf_sent_cb, (void *)id);
+ }
+ }
+}
+
+static int bearer_ctl_send(int i, u8_t op, void *data, u8_t data_len)
+{
+ struct net_buf *buf;
+
+ BT_DBG("op 0x%02x data_len %u", op, data_len);
+
+ prov_clear_tx(i);
+
+ buf = adv_buf_create();
+ if (!buf) {
+ return -ENOBUFS;
+ }
+
+ net_buf_add_be32(buf, link[i].link_id);
+ /* Transaction ID, always 0 for Bearer messages */
+ net_buf_add_u8(buf, 0x00);
+ net_buf_add_u8(buf, GPC_CTL(op));
+ net_buf_add_mem(buf, data, data_len);
+
+ link[i].tx.buf[0] = buf;
+ send_reliable(i);
+
+ /** We can also use buf->ref and a flag to decide that
+ * link close has been sent 3 times.
+ * Here we use another way: use retransmit timer and need
+ * to make sure the timer is not cancelled during sending
+ * link close pdu, so we add link[i].tx.id = 0
+ */
+ if (op == LINK_CLOSE) {
+ u8_t reason = *(u8_t *)data;
+ link[i].link_close = (reason << 8 | BIT(0));
+ link[i].tx.trans_id = 0;
+ }
+
+ return 0;
+}
+
+static void send_link_open(void)
+{
+ int i = prov_get_pb_index(), j;
+
+ /** Generate link ID, and may need to check if this id is
+ * currently being used, which may will not happen ever.
+ */
+ bt_mesh_rand(&link[i].link_id, sizeof(u32_t));
+ while (1) {
+ for (j = 0; j < CONFIG_BT_MESH_PBA_SAME_TIME; j++) {
+ if (atomic_test_bit(link[j].flags, LINK_ACTIVE) || link[j].linking) {
+ if (link[i].link_id == link[j].link_id) {
+ bt_mesh_rand(&link[i].link_id, sizeof(u32_t));
+ break;
+ }
+ }
+ }
+ if (j == CONFIG_BT_MESH_PBA_SAME_TIME) {
+ break;
+ }
+ }
+
+ bearer_ctl_send(i, LINK_OPEN, link[i].uuid, 16);
+
+ /* Set LINK_ACTIVE just to be in compatibility with current Zephyr code */
+ atomic_set_bit(link[i].flags, LINK_ACTIVE);
+
+ prov_ctx.pba_count++;
+}
+
+static u8_t last_seg(u8_t len)
+{
+ if (len <= START_PAYLOAD_MAX) {
+ return 0;
+ }
+
+ len -= START_PAYLOAD_MAX;
+
+ return 1 + (len / CONT_PAYLOAD_MAX);
+}
+
+static inline u8_t next_transaction_id(void)
+{
+ int i = prov_get_pb_index();
+
+ if (link[i].tx.trans_id < 0x7F) {
+ return link[i].tx.trans_id++;
+ }
+
+ return 0x0;
+}
+
+static int prov_send_adv(struct net_buf_simple *msg)
+{
+ struct net_buf *start, *buf;
+ u8_t seg_len, seg_id;
+ u8_t xact_id;
+ int i = prov_get_pb_index();
+
+ BT_DBG("%s, len %u: %s", __func__, msg->len, bt_hex(msg->data, msg->len));
+
+ prov_clear_tx(i);
+
+ start = adv_buf_create();
+ if (!start) {
+ return -ENOBUFS;
+ }
+
+ xact_id = next_transaction_id();
+ net_buf_add_be32(start, link[i].link_id);
+ net_buf_add_u8(start, xact_id);
+
+ net_buf_add_u8(start, GPC_START(last_seg(msg->len)));
+ net_buf_add_be16(start, msg->len);
+ net_buf_add_u8(start, bt_mesh_fcs_calc(msg->data, msg->len));
+
+ link[i].tx.buf[0] = start;
+
+ seg_len = min(msg->len, START_PAYLOAD_MAX);
+ BT_DBG("seg 0 len %u: %s", seg_len, bt_hex(msg->data, seg_len));
+ net_buf_add_mem(start, msg->data, seg_len);
+ net_buf_simple_pull(msg, seg_len);
+
+ buf = start;
+ for (seg_id = 1; msg->len > 0; seg_id++) {
+ if (seg_id >= ARRAY_SIZE(link[i].tx.buf)) {
+ BT_ERR("%s, Too big message", __func__);
+ free_segments(i);
+ return -E2BIG;
+ }
+
+ buf = adv_buf_create();
+ if (!buf) {
+ free_segments(i);
+ return -ENOBUFS;
+ }
+
+ link[i].tx.buf[seg_id] = buf;
+
+ seg_len = min(msg->len, CONT_PAYLOAD_MAX);
+
+ BT_DBG("seg_id %u len %u: %s", seg_id, seg_len,
+ bt_hex(msg->data, seg_len));
+
+ net_buf_add_be32(buf, link[i].link_id);
+ net_buf_add_u8(buf, xact_id);
+ net_buf_add_u8(buf, GPC_CONT(seg_id));
+ net_buf_add_mem(buf, msg->data, seg_len);
+ net_buf_simple_pull(msg, seg_len);
+ }
+
+ send_reliable(i);
+
+ if (!atomic_test_and_set_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_submit(&link[i].timeout, PROVISION_TIMEOUT);
+ }
+
+ return 0;
+}
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+static int prov_send_gatt(struct net_buf_simple *msg)
+{
+ int i = prov_get_pb_index();
+ int err;
+
+ if (!link[i].conn) {
+ return -ENOTCONN;
+ }
+
+ err = provisioner_proxy_send(link[i].conn, BT_MESH_PROXY_PROV, msg);
+ if (err) {
+ BT_ERR("%s, Failed to send PB-GATT pdu", __func__);
+ return err;
+ }
+
+ if (!atomic_test_and_set_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_submit(&link[i].timeout, PROVISION_TIMEOUT);
+ }
+
+ return 0;
+}
+#endif /* CONFIG_BT_MESH_PB_GATT */
+
+static inline int prov_send(struct net_buf_simple *buf)
+{
+ int i = prov_get_pb_index();
+
+ if (i < CONFIG_BT_MESH_PBA_SAME_TIME) {
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ return prov_send_adv(buf);
+#else
+ return -EINVAL;
+#endif
+ } else if (i >= CONFIG_BT_MESH_PBA_SAME_TIME &&
+ i < BT_MESH_PROV_SAME_TIME) {
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ return prov_send_gatt(buf);
+#else
+ return -EINVAL;
+#endif
+ } else {
+ BT_ERR("%s, Link index exceeding upper limit", __func__);
+ return -EINVAL;
+ }
+}
+
+static void prov_buf_init(struct net_buf_simple *buf, u8_t type)
+{
+ net_buf_simple_init(buf, PROV_BUF_HEADROOM);
+ net_buf_simple_add_u8(buf, type);
+}
+
+static void prov_invite(const u8_t *data)
+{
+ BT_DBG("%s", __func__);
+}
+
+static void prov_start(const u8_t *data)
+{
+ BT_DBG("%s", __func__);
+}
+
+static void prov_data(const u8_t *data)
+{
+ BT_DBG("%s", __func__);
+}
+
+static void send_invite(void)
+{
+ struct net_buf_simple *buf = PROV_BUF(2);
+ int i = prov_get_pb_index();
+
+ prov_buf_init(buf, PROV_INVITE);
+
+ net_buf_simple_add_u8(buf, prov->prov_attention);
+
+ link[i].conf_inputs[0] = prov->prov_attention;
+
+ if (prov_send(buf)) {
+ BT_ERR("%s, Failed to send Provisioning Invite", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ link[i].expect = PROV_CAPABILITIES;
+}
+
+static void prov_capabilities(const u8_t *data)
+{
+ struct net_buf_simple *buf = PROV_BUF(6);
+ u16_t algorithms, output_action, input_action;
+ u8_t element_num, pub_key_oob, static_oob,
+ output_size, input_size;
+ u8_t auth_method, auth_action, auth_size;
+ int i = prov_get_pb_index();
+
+ element_num = data[0];
+ BT_DBG("Elements: %u", element_num);
+ if (!element_num) {
+ BT_ERR("%s, Invalid element number", __func__);
+ goto fail;
+ }
+ link[i].element_num = element_num;
+
+ algorithms = sys_get_be16(&data[1]);
+ BT_DBG("Algorithms: %u", algorithms);
+ if (algorithms != BIT(PROV_ALG_P256)) {
+ BT_ERR("%s, Invalid algorithms", __func__);
+ goto fail;
+ }
+
+ pub_key_oob = data[3];
+ BT_DBG("Public Key Type: 0x%02x", pub_key_oob);
+ if (pub_key_oob > 0x01) {
+ BT_ERR("%s, Invalid public key type", __func__);
+ goto fail;
+ }
+ pub_key_oob = ((prov->prov_pub_key_oob &&
+ prov->prov_pub_key_oob_cb) ? pub_key_oob : 0x00);
+
+ static_oob = data[4];
+ BT_DBG("Static OOB Type: 0x%02x", static_oob);
+ if (static_oob > 0x01) {
+ BT_ERR("%s, Invalid Static OOB type", __func__);
+ goto fail;
+ }
+ static_oob = (prov->prov_static_oob_val ? static_oob : 0x00);
+
+ output_size = data[5];
+ BT_DBG("Output OOB Size: %u", output_size);
+ if (output_size > 0x08) {
+ BT_ERR("%s, Invalid Output OOB size", __func__);
+ goto fail;
+ }
+
+ output_action = sys_get_be16(&data[6]);
+ BT_DBG("Output OOB Action: 0x%04x", output_action);
+ if (output_action > 0x1f) {
+ BT_ERR("%s, Invalid Output OOB action", __func__);
+ goto fail;
+ }
+
+ /* Provisioner select output action */
+ if (prov->prov_input_num && output_size) {
+ output_action = __builtin_ctz(output_action);
+ } else {
+ output_size = 0x0;
+ output_action = 0x0;
+ }
+
+ input_size = data[8];
+ BT_DBG("Input OOB Size: %u", input_size);
+ if (input_size > 0x08) {
+ BT_ERR("%s, Invalid Input OOB size", __func__);
+ goto fail;
+ }
+
+ input_action = sys_get_be16(&data[9]);
+ BT_DBG("Input OOB Action: 0x%04x", input_action);
+ if (input_action > 0x0f) {
+ BT_ERR("%s, Invalid Input OOB action", __func__);
+ goto fail;
+ }
+
+ /* Make sure received pdu is ok and cancel the timeout timer */
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ /* Provisioner select input action */
+ if (prov->prov_output_num && input_size) {
+ input_action = __builtin_ctz(input_action);
+ } else {
+ input_size = 0x0;
+ input_action = 0x0;
+ }
+
+ if (static_oob) {
+ /* if static oob is valid, just use static oob */
+ auth_method = AUTH_METHOD_STATIC;
+ auth_action = 0x00;
+ auth_size = 0x00;
+ } else {
+ if (!output_size && !input_size) {
+ auth_method = AUTH_METHOD_NO_OOB;
+ auth_action = 0x00;
+ auth_size = 0x00;
+ } else if (!output_size && input_size) {
+ auth_method = AUTH_METHOD_INPUT;
+ auth_action = (u8_t)input_action;
+ auth_size = input_size;
+ } else {
+ auth_method = AUTH_METHOD_OUTPUT;
+ auth_action = (u8_t)output_action;
+ auth_size = output_size;
+ }
+ }
+
+ /* Store provisioning capbilities value in conf_inputs */
+ memcpy(&link[i].conf_inputs[1], data, 11);
+
+ prov_buf_init(buf, PROV_START);
+ net_buf_simple_add_u8(buf, prov->prov_algorithm);
+ net_buf_simple_add_u8(buf, pub_key_oob);
+ net_buf_simple_add_u8(buf, auth_method);
+ net_buf_simple_add_u8(buf, auth_action);
+ net_buf_simple_add_u8(buf, auth_size);
+
+ memcpy(&link[i].conf_inputs[12], &buf->data[1], 5);
+
+ if (prov_send(buf)) {
+ BT_ERR("%s, Failed to send Provisioning Start", __func__);
+ goto fail;
+ }
+
+ link[i].auth_method = auth_method;
+ link[i].auth_action = auth_action;
+ link[i].auth_size = auth_size;
+
+ /** After prov start sent, use OOB to get remote public key.
+ * And we just follow the procedure in Figure 5.15 of Section
+ * 5.4.2.3 of Mesh Profile Spec.
+ */
+ if (pub_key_oob) {
+ if (prov->prov_pub_key_oob_cb(i)) {
+ BT_ERR("%s, Failed to notify input OOB Public Key", __func__);
+ goto fail;
+ }
+ }
+
+ /** If using PB-ADV, need to listen for transaction ack,
+ * after ack is received, provisioner can send public key.
+ */
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ if (i < CONFIG_BT_MESH_PBA_SAME_TIME) {
+ link[i].expect_ack_for = PROV_START;
+ return;
+ }
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+ send_pub_key(pub_key_oob);
+ return;
+
+fail:
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+}
+
+static bt_mesh_output_action_t output_action(u8_t action)
+{
+ switch (action) {
+ case OUTPUT_OOB_BLINK:
+ return BT_MESH_BLINK;
+ case OUTPUT_OOB_BEEP:
+ return BT_MESH_BEEP;
+ case OUTPUT_OOB_VIBRATE:
+ return BT_MESH_VIBRATE;
+ case OUTPUT_OOB_NUMBER:
+ return BT_MESH_DISPLAY_NUMBER;
+ case OUTPUT_OOB_STRING:
+ return BT_MESH_DISPLAY_STRING;
+ default:
+ return BT_MESH_NO_OUTPUT;
+ }
+}
+
+static bt_mesh_input_action_t input_action(u8_t action)
+{
+ switch (action) {
+ case INPUT_OOB_PUSH:
+ return BT_MESH_PUSH;
+ case INPUT_OOB_TWIST:
+ return BT_MESH_TWIST;
+ case INPUT_OOB_NUMBER:
+ return BT_MESH_ENTER_NUMBER;
+ case INPUT_OOB_STRING:
+ return BT_MESH_ENTER_STRING;
+ default:
+ return BT_MESH_NO_INPUT;
+ }
+}
+
+static int prov_auth(u8_t method, u8_t action, u8_t size)
+{
+ bt_mesh_output_action_t output;
+ bt_mesh_input_action_t input;
+ int i = prov_get_pb_index();
+
+ link[i].auth = (u8_t *)osi_calloc(PROV_AUTH_VAL_SIZE);
+ if (!link[i].auth) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return -ENOMEM;
+ }
+
+ switch (method) {
+ case AUTH_METHOD_NO_OOB:
+ if (action || size) {
+ return -EINVAL;
+ }
+ memset(link[i].auth, 0, 16);
+ return 0;
+
+ case AUTH_METHOD_STATIC:
+ if (action || size) {
+ return -EINVAL;
+ }
+ memcpy(link[i].auth + 16 - prov->prov_static_oob_len,
+ prov->prov_static_oob_val, prov->prov_static_oob_len);
+ memset(link[i].auth, 0, 16 - prov->prov_static_oob_len);
+ return 0;
+
+ case AUTH_METHOD_OUTPUT:
+ /* Use auth_action to get device output action */
+ output = output_action(action);
+ if (!output) {
+ return -EINVAL;
+ }
+ return prov->prov_input_num(AUTH_METHOD_OUTPUT, output, size, i);
+
+ case AUTH_METHOD_INPUT:
+ /* Use auth_action to get device input action */
+ input = input_action(action);
+ if (!input) {
+ return -EINVAL;
+ }
+
+ /* Provisioner ouputs number/string and wait for device's Provisioning Input Complete PDU */
+ link[i].expect = PROV_INPUT_COMPLETE;
+
+ if (input == BT_MESH_ENTER_STRING) {
+ unsigned char str[9];
+ u8_t j;
+
+ bt_mesh_rand(str, size);
+ /* Normalize to '0' .. '9' & 'A' .. 'Z' */
+ for (j = 0; j < size; j++) {
+ str[j] %= 36;
+ if (str[j] < 10) {
+ str[j] += '0';
+ } else {
+ str[j] += 'A' - 10;
+ }
+ }
+ str[size] = '\0';
+
+ memcpy(link[i].auth, str, size);
+ memset(link[i].auth + size, 0, sizeof(link[i].auth) - size);
+
+ return prov->prov_output_num(AUTH_METHOD_INPUT, input, str, size, i);
+ } else {
+ u32_t div[8] = { 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000 };
+ u32_t num;
+
+ bt_mesh_rand(&num, sizeof(num));
+ num %= div[size - 1];
+
+ sys_put_be32(num, &link[i].auth[12]);
+ memset(link[i].auth, 0, 12);
+
+ return prov->prov_output_num(AUTH_METHOD_INPUT, input, &num, size, i);
+ }
+
+ default:
+ return -EINVAL;
+ }
+}
+
+static void send_confirm(void)
+{
+ struct net_buf_simple *buf = PROV_BUF(17);
+ int i = prov_get_pb_index();
+
+ BT_DBG("ConfInputs[0] %s", bt_hex(link[i].conf_inputs, 64));
+ BT_DBG("ConfInputs[64] %s", bt_hex(link[i].conf_inputs + 64, 64));
+ BT_DBG("ConfInputs[128] %s", bt_hex(link[i].conf_inputs + 128, 17));
+
+ link[i].conf_salt = (u8_t *)osi_calloc(PROV_CONF_SALT_SIZE);
+ if (!link[i].conf_salt) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ goto fail;
+ }
+
+ link[i].conf_key = (u8_t *)osi_calloc(PROV_CONF_KEY_SIZE);
+ if (!link[i].conf_key) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ goto fail;
+ }
+
+ if (bt_mesh_prov_conf_salt(link[i].conf_inputs, link[i].conf_salt)) {
+ BT_ERR("%s, Failed to generate confirmation salt", __func__);
+ goto fail;
+ }
+
+ BT_DBG("ConfirmationSalt: %s", bt_hex(link[i].conf_salt, 16));
+
+ if (bt_mesh_prov_conf_key(link[i].dhkey, link[i].conf_salt, link[i].conf_key)) {
+ BT_ERR("%s, Failed to generate confirmation key", __func__);
+ goto fail;
+ }
+
+ BT_DBG("ConfirmationKey: %s", bt_hex(link[i].conf_key, 16));
+
+ /** Provisioner use the same random number for each provisioning
+ * device, if different random need to be used, here provisioner
+ * should allocate memory for rand and call bt_mesh_rand() every time.
+ */
+ if (!(prov_ctx.rand_gen_done & BIT(0))) {
+ if (bt_mesh_rand(prov_ctx.random, 16)) {
+ BT_ERR("%s, Failed to generate random number", __func__);
+ goto fail;
+ }
+ link[i].rand = prov_ctx.random;
+ prov_ctx.rand_gen_done |= BIT(0);
+ } else {
+ /* Provisioner random has already been generated. */
+ link[i].rand = prov_ctx.random;
+ }
+
+ BT_DBG("LocalRandom: %s", bt_hex(link[i].rand, 16));
+
+ prov_buf_init(buf, PROV_CONFIRM);
+
+ if (bt_mesh_prov_conf(link[i].conf_key, link[i].rand, link[i].auth,
+ net_buf_simple_add(buf, 16))) {
+ BT_ERR("%s, Failed to generate confirmation value", __func__);
+ goto fail;
+ }
+
+ if (prov_send(buf)) {
+ BT_ERR("%s, Failed to send Provisioning Confirm", __func__);
+ goto fail;
+ }
+
+ link[i].expect = PROV_CONFIRM;
+ return;
+
+fail:
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+}
+
+int bt_mesh_prov_set_oob_input_data(u8_t *val, u8_t link_idx, bool num_flag)
+{
+ /** This function should be called in the prov_input_num
+ * callback, after the data output by device has been
+ * input by provisioner.
+ * Paramter size is used to indicate the length of data
+ * indicated by Pointer val, for example, if device output
+ * data is 12345678(decimal), the data in auth value will
+ * be 0xBC614E.
+ * Parameter num_flag is used to indicate whether the value
+ * input by provisioner is number or string.
+ */
+ if (!link[link_idx].auth) {
+ BT_ERR("%s, Link auth is NULL", __func__);
+ return -EINVAL;
+ }
+
+ memset(link[link_idx].auth, 0, 16);
+ if (num_flag) {
+ /* Provisioner inputs number */
+ memcpy(link[link_idx].auth + 12, val, sizeof(u32_t));
+ } else {
+ /* Provisioner inputs string */
+ memcpy(link[link_idx].auth, val, link[link_idx].auth_size);
+ }
+
+ prov_set_pb_index(link_idx);
+ send_confirm();
+
+ return 0;
+}
+
+// int bt_mesh_prov_set_oob_output_data(u8_t *num, u8_t size, bool num_flag, u8_t link_idx)
+// {
+// /** This function should be called in the prov_output_num
+// * callback, after the data has been output by provisioner.
+// * Parameter size is used to indicate the length of data
+// * indicated by Pointer num, for example, if provisioner
+// * output data is 12345678(decimal), the data in auth value
+// * will be 0xBC614E.
+// * Parameter num_flag is used to indicate whether the value
+// * output by provisioner is number or string.
+// */
+// if (!link[link_idx].auth) {
+// BT_ERR("%s, link auth is NULL", __func__);
+// return -EINVAL;
+// }
+
+// if (num_flag) {
+// /* Provisioner output number */
+// memset(link[link_idx].auth, 0, 16);
+// memcpy(link[link_idx].auth + 16 - size, num, size);
+// } else {
+// /* Provisioner output string */
+// memset(link[link_idx].auth, 0, 16);
+// memcpy(link[link_idx].auth, num, size);
+// }
+
+// link[link_idx].expect = PROV_INPUT_COMPLETE;
+
+// return 0;
+// }
+
+int bt_mesh_prov_read_oob_pub_key(u8_t link_idx, u8_t pub_key_x[32], u8_t pub_key_y[32])
+{
+ if (!link[link_idx].conf_inputs) {
+ BT_ERR("%s, Link conf_inputs is NULL", __func__);
+ return -EINVAL;
+ }
+
+ /* Swap X and Y halves independently to big-endian */
+ sys_memcpy_swap(&link[link_idx].conf_inputs[81], pub_key_x, 32);
+ sys_memcpy_swap(&link[link_idx].conf_inputs[81] + 32, pub_key_y, 32);
+
+ atomic_set_bit(link[link_idx].flags, REMOTE_PUB_KEY);
+
+ if (atomic_test_and_clear_bit(link[link_idx].flags, WAIT_GEN_DHKEY)) {
+ prov_gen_dh_key(link_idx);
+ }
+
+ return 0;
+}
+
+static void prov_dh_key_cb(const u8_t key[32])
+{
+ int i = prov_get_pb_index();
+
+ BT_DBG("%p", key);
+
+ if (!key) {
+ BT_ERR("%s, Failed to generate DHKey", __func__);
+ goto fail;
+ }
+
+ link[i].dhkey = (u8_t *)osi_calloc(PROV_DH_KEY_SIZE);
+ if (!link[i].dhkey) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ goto fail;
+ }
+ sys_memcpy_swap(link[i].dhkey, key, 32);
+
+ BT_DBG("DHkey: %s", bt_hex(link[i].dhkey, 32));
+
+ atomic_set_bit(link[i].flags, HAVE_DHKEY);
+
+ /** After dhkey is generated, if auth_method is No OOB or
+ * Static OOB, provisioner can start to send confirmation.
+ * If output OOB is used by the device, provisioner need
+ * to watch out the output number and input it as auth_val.
+ * If input OOB is used by the device, provisioner need
+ * to output a value, and wait for prov input complete pdu.
+ */
+ if (prov_auth(link[i].auth_method,
+ link[i].auth_action, link[i].auth_size) < 0) {
+ BT_ERR("%s, Failed to authenticate", __func__);
+ goto fail;
+ }
+ if (link[i].auth_method == AUTH_METHOD_OUTPUT ||
+ link[i].auth_method == AUTH_METHOD_INPUT) {
+ return;
+ }
+
+ if (link[i].expect != PROV_INPUT_COMPLETE) {
+ send_confirm();
+ }
+ return;
+
+fail:
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+}
+
+static void prov_gen_dh_key(int i)
+{
+ u8_t pub_key[64];
+
+ /* Copy device public key in little-endian for bt_mesh_dh_key_gen().
+ * X and Y halves are swapped independently.
+ */
+ sys_memcpy_swap(&pub_key[0], &link[i].conf_inputs[81], 32);
+ sys_memcpy_swap(&pub_key[32], &link[i].conf_inputs[113], 32);
+
+ if (bt_mesh_dh_key_gen(pub_key, prov_dh_key_cb)) {
+ BT_ERR("%s, Failed to generate DHKey", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+}
+
+static void send_pub_key(u8_t oob)
+{
+ struct net_buf_simple *buf = PROV_BUF(65);
+ const u8_t *key = NULL;
+ int i = prov_get_pb_index();
+
+ key = bt_mesh_pub_key_get();
+ if (!key) {
+ BT_ERR("%s, No public key available", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ BT_DBG("Local Public Key: %s", bt_hex(key, 64));
+
+ atomic_set_bit(link[i].flags, LOCAL_PUB_KEY);
+
+ prov_buf_init(buf, PROV_PUB_KEY);
+
+ /* Swap X and Y halves independently to big-endian */
+ sys_memcpy_swap(net_buf_simple_add(buf, 32), key, 32);
+ sys_memcpy_swap(net_buf_simple_add(buf, 32), &key[32], 32);
+
+ /* Store provisioner public key value in conf_inputs */
+ memcpy(&link[i].conf_inputs[17], &buf->data[1], 64);
+
+ if (prov_send(buf)) {
+ BT_ERR("%s, Failed to send Provisioning Public Key", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (!oob) {
+ link[i].expect = PROV_PUB_KEY;
+ } else {
+ /** Have already got device public key. If next is to
+ * send confirm(not wait for input complete), need to
+ * wait for transactiona ack for public key then send
+ * provisioning confirm pdu.
+ */
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ if (i < CONFIG_BT_MESH_PBA_SAME_TIME) {
+ link[i].expect_ack_for = PROV_PUB_KEY;
+ return;
+ }
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+ /* If remote public key has been read, then start to generate DHkey,
+ * otherwise wait for device oob public key.
+ */
+ if (atomic_test_bit(link[i].flags, REMOTE_PUB_KEY)) {
+ prov_gen_dh_key(i);
+ } else {
+ atomic_set_bit(link[i].flags, WAIT_GEN_DHKEY);
+ }
+ }
+}
+
+static void prov_pub_key(const u8_t *data)
+{
+ int i = prov_get_pb_index();
+
+ BT_DBG("Remote Public Key: %s", bt_hex(data, 64));
+
+ /* Make sure received pdu is ok and cancel the timeout timer */
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ memcpy(&link[i].conf_inputs[81], data, 64);
+
+ if (!atomic_test_bit(link[i].flags, LOCAL_PUB_KEY)) {
+ /* Clear retransmit timer */
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ prov_clear_tx(i);
+#endif
+ atomic_set_bit(link[i].flags, REMOTE_PUB_KEY);
+ BT_WARN("%s, Waiting for local public key", __func__);
+ return;
+ }
+
+ prov_gen_dh_key(i);
+}
+
+static void prov_input_complete(const u8_t *data)
+{
+ int i = prov_get_pb_index();
+
+ /* Make sure received pdu is ok and cancel the timeout timer */
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ /* Provisioner receives input complete and send confirm */
+ send_confirm();
+}
+
+static void prov_confirm(const u8_t *data)
+{
+ /** Here Zephyr uses PROV_BUF(16). Currently test with PROV_BUF(16)
+ * and PROV_BUF(17) on branch feature/btdm_ble_mesh_debug both
+ * work fine.
+ */
+ struct net_buf_simple *buf = PROV_BUF(17);
+ int i = prov_get_pb_index();
+
+ BT_DBG("Remote Confirm: %s", bt_hex(data, 16));
+
+ /* Make sure received pdu is ok and cancel the timeout timer */
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ link[i].conf = (u8_t *)osi_calloc(PROV_CONFIRM_SIZE);
+ if (!link[i].conf) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ memcpy(link[i].conf, data, 16);
+
+ if (!atomic_test_bit(link[i].flags, HAVE_DHKEY)) {
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ prov_clear_tx(i);
+#endif
+ atomic_set_bit(link[i].flags, SEND_CONFIRM);
+ }
+
+ prov_buf_init(buf, PROV_RANDOM);
+
+ net_buf_simple_add_mem(buf, link[i].rand, 16);
+
+ if (prov_send(buf)) {
+ BT_ERR("%s, Failed to send Provisioning Random", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ link[i].expect = PROV_RANDOM;
+}
+
+static void send_prov_data(void)
+{
+ struct net_buf_simple *buf = PROV_BUF(34);
+ const u8_t *netkey = NULL;
+ int i = prov_get_pb_index();
+ int j, err;
+ bool already_flag = false;
+ u8_t session_key[16];
+ u8_t nonce[13];
+ u8_t pdu[25];
+ u16_t max_addr;
+
+ err = bt_mesh_session_key(link[i].dhkey, link[i].prov_salt, session_key);
+ if (err) {
+ BT_ERR("%s, Failed to generate session key", __func__);
+ goto fail;
+ }
+ BT_DBG("SessionKey: %s", bt_hex(session_key, 16));
+
+ err = bt_mesh_prov_nonce(link[i].dhkey, link[i].prov_salt, nonce);
+ if (err) {
+ BT_ERR("%s, Failed to generate session nonce", __func__);
+ goto fail;
+ }
+ BT_DBG("Nonce: %s", bt_hex(nonce, 13));
+
+ /* Assign provisioning data for the device. Currently all provisioned devices
+ * will be added to the primary subnet, and may add an API to choose to which
+ * subnet will the device be provisioned later.
+ */
+ if (FAST_PROV_FLAG_GET()) {
+ netkey = fast_prov.net_key;
+ if (!netkey) {
+ BT_ERR("%s, Failed to get NetKey for fast provisioning", __func__);
+ goto fail;
+ }
+ memcpy(pdu, netkey, 16);
+ sys_put_be16(fast_prov.net_idx, &pdu[16]);
+ pdu[18] = fast_prov.flags;
+ sys_put_be32(fast_prov.iv_index, &pdu[19]);
+ } else {
+ netkey = provisioner_net_key_get(prov_ctx.curr_net_idx);
+ if (!netkey) {
+ BT_ERR("%s, Failed to get NetKey for provisioning data", __func__);
+ goto fail;
+ }
+ memcpy(pdu, netkey, 16);
+ sys_put_be16(prov_ctx.curr_net_idx, &pdu[16]);
+ pdu[18] = prov_ctx.curr_flags;
+ sys_put_be32(prov_ctx.curr_iv_index, &pdu[19]);
+ }
+
+ /* 1. The Provisioner must not reuse unicast addresses that have been
+ * allocated to a device and sent in a Provisioning Data PDU until
+ * the Provisioner receives an Unprovisioned Device beacon or
+ * Service Data for the Mesh Provisioning Service from that same
+ * device, identified using the Device UUID of the device.
+ * 2. Once the provisioning data for the device has been sent, we will
+ * add the data sent to this device into the already_prov_info.
+ * 3. Another situation here is:
+ * If the device is a re-provisioned one, but the element num has
+ * changed and is larger than the previous number, here we will
+ * assign new address for the device.
+ */
+
+ /* Check if this device is a re-provisioned device */
+ for (j = 0; j < BT_MESH_ALREADY_PROV_NUM; j++) {
+ if (!memcmp(link[i].uuid, prov_ctx.already_prov[j].uuid, 16)) {
+ if (link[i].element_num <= prov_ctx.already_prov[j].element_num) {
+ already_flag = true;
+ sys_put_be16(prov_ctx.already_prov[j].unicast_addr, &pdu[23]);
+ link[i].unicast_addr = prov_ctx.already_prov[j].unicast_addr;
+ break;
+ } else {
+ /* TODO: If the device has a larger element number during the
+ * second provisioning, then if the device is provisioned the
+ * third time later, already_prov struct will have two elements
+ * containing the same device UUID but with different element
+ * number. So we may add a flag to indicate the unicast address
+ * in the smaller element can be reused by other devices when
+ * unicast address is exhausted.
+ */
+ }
+ }
+ }
+
+ max_addr = FAST_PROV_FLAG_GET() ? fast_prov.unicast_addr_max : 0x7FFF;
+
+ if (!already_flag) {
+ /* If this device to be provisioned is a new device */
+ if (!prov_ctx.current_addr) {
+ BT_ERR("%s, No unicast address can be assigned", __func__);
+ goto fail;
+ }
+
+ if (prov_ctx.current_addr + link[i].element_num - 1 > max_addr) {
+ BT_ERR("%s, Not enough unicast address for the device", __func__);
+ goto fail;
+ }
+
+ sys_put_be16(prov_ctx.current_addr, &pdu[23]);
+ link[i].unicast_addr = prov_ctx.current_addr;
+ }
+
+ prov_buf_init(buf, PROV_DATA);
+
+ err = bt_mesh_prov_encrypt(session_key, nonce, pdu, net_buf_simple_add(buf, 33));
+ if (err) {
+ BT_ERR("%s, Failed to encrypt provisioning data", __func__);
+ goto fail;
+ }
+
+ if (prov_send(buf)) {
+ BT_ERR("%s, Failed to send Provisioning Data", __func__);
+ goto fail;
+ }
+
+ /* If provisioning data is sent successfully, add the assigned information
+ * into the already_prov_info struct if this device is a new one. And if
+ * sent successfully, update the current_addr in prov_ctx struct.
+ */
+ if (!already_flag) {
+ for (j = 0; j < BT_MESH_ALREADY_PROV_NUM; j++) {
+ if (!prov_ctx.already_prov[j].element_num) {
+ memcpy(prov_ctx.already_prov[j].uuid, link[i].uuid, 16);
+ prov_ctx.already_prov[j].element_num = link[i].element_num;
+ prov_ctx.already_prov[j].unicast_addr = link[i].unicast_addr;
+ break;
+ }
+ }
+
+ /* We update the next unicast address to be assigned here because
+ * if provisioner is provisioning two devices at the same time, we
+ * need to assign the unicast address for them correctly. Hence we
+ * should not update the prov_ctx.current_addr after the proper
+ * provisioning complete pdu is received.
+ */
+ prov_ctx.current_addr += link[i].element_num;
+ if (prov_ctx.current_addr > max_addr) {
+ /* No unicast address will be used for further provisioning */
+ prov_ctx.current_addr = 0x0000;
+ }
+ }
+
+ if (FAST_PROV_FLAG_GET()) {
+ link[i].ki_flags = fast_prov.flags;
+ link[i].iv_index = fast_prov.iv_index;
+ } else {
+ link[i].ki_flags = prov_ctx.curr_flags;
+ link[i].iv_index = prov_ctx.curr_iv_index;
+ }
+
+ link[i].expect = PROV_COMPLETE;
+ return;
+
+fail:
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+}
+
+static void prov_random(const u8_t *data)
+{
+ u8_t conf_verify[16];
+ int i = prov_get_pb_index();
+
+ BT_DBG("Remote Random: %s", bt_hex(data, 16));
+
+ if (bt_mesh_prov_conf(link[i].conf_key, data, link[i].auth, conf_verify)) {
+ BT_ERR("%s, Failed to calculate confirmation verification", __func__);
+ goto fail;
+ }
+
+ if (memcmp(conf_verify, link[i].conf, 16)) {
+ BT_ERR("%s, Invalid confirmation value", __func__);
+ BT_DBG("Received: %s", bt_hex(link[i].conf, 16));
+ BT_DBG("Calculated: %s", bt_hex(conf_verify, 16));
+ goto fail;
+ }
+
+ /*Verify received confirm is ok and cancel the timeout timer */
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ /** After provisioner receives provisioning random from device,
+ * and successfully check the confirmation, the following
+ * should be done:
+ * 1. osi_calloc memory for prov_salt
+ * 2. calculate prov_salt
+ * 3. prepare provisioning data and send
+ */
+ link[i].prov_salt = (u8_t *)osi_calloc(PROV_PROV_SALT_SIZE);
+ if (!link[i].prov_salt) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ goto fail;
+ }
+
+ if (bt_mesh_prov_salt(link[i].conf_salt, link[i].rand, data,
+ link[i].prov_salt)) {
+ BT_ERR("%s, Failed to generate ProvisioningSalt", __func__);
+ goto fail;
+ }
+
+ BT_DBG("ProvisioningSalt: %s", bt_hex(link[i].prov_salt, 16));
+
+ send_prov_data();
+ return;
+
+fail:
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+}
+
+static void prov_complete(const u8_t *data)
+{
+ u8_t device_key[16];
+ int i = prov_get_pb_index(), j;
+ int err, rm = 0;
+
+ /* Make sure received pdu is ok and cancel the timeout timer */
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ /* If provisioning complete is received, the provisioning device
+ * will be stored into the prov_node_info structure and become a
+ * node within the mesh network
+ */
+ err = bt_mesh_dev_key(link[i].dhkey, link[i].prov_salt, device_key);
+ if (err) {
+ BT_ERR("%s, Failed to generate device key", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ for (j = 0; j < ARRAY_SIZE(prov_nodes); j++) {
+ if (!prov_nodes[j].provisioned) {
+ prov_nodes[j].provisioned = true;
+ prov_nodes[j].oob_info = link[i].oob_info;
+ prov_nodes[j].element_num = link[i].element_num;
+ prov_nodes[j].unicast_addr = link[i].unicast_addr;
+ if (FAST_PROV_FLAG_GET()) {
+ prov_nodes[j].net_idx = fast_prov.net_idx;
+ } else {
+ prov_nodes[j].net_idx = prov_ctx.curr_net_idx;
+ }
+ prov_nodes[j].flags = link[i].ki_flags;
+ prov_nodes[j].iv_index = link[i].iv_index;
+ prov_nodes[j].addr.type = link[i].addr.type;
+ memcpy(prov_nodes[j].addr.a.val, link[i].addr.a.val, BD_ADDR_LEN);
+ memcpy(prov_nodes[j].uuid, link[i].uuid, 16);
+ break;
+ }
+ }
+
+ if (j == ARRAY_SIZE(prov_nodes)) {
+ BT_ERR("%s, Provisioned node queue is full", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ prov_ctx.node_count++;
+
+ err = provisioner_node_provision(j, prov_nodes[j].uuid, prov_nodes[j].oob_info,
+ prov_nodes[j].unicast_addr, prov_nodes[j].element_num,
+ prov_nodes[j].net_idx, prov_nodes[j].flags,
+ prov_nodes[j].iv_index, device_key);
+ if (err) {
+ BT_ERR("%s, Failed to store node info", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (prov->prov_complete) {
+ prov->prov_complete(j, prov_nodes[j].uuid, prov_nodes[j].unicast_addr,
+ prov_nodes[j].element_num, prov_nodes[j].net_idx);
+ }
+
+ err = provisioner_dev_find(&link[i].addr, link[i].uuid, &rm);
+ if (!err) {
+ if (unprov_dev[rm].flags & RM_AFTER_PROV) {
+ memset(&unprov_dev[rm], 0, sizeof(struct unprov_dev_queue));
+ }
+ } else if (err == -ENODEV) {
+ BT_DBG("%s, Device is not found in queue", __func__);
+ } else {
+ BT_WARN("%s, Failed to remove device from queue", __func__);
+ }
+
+ close_link(i, CLOSE_REASON_SUCCESS);
+}
+
+static void prov_failed(const u8_t *data)
+{
+ int i = prov_get_pb_index();
+
+ BT_WARN("%s, Error 0x%02x", __func__, data[0]);
+
+ close_link(i, CLOSE_REASON_FAILED);
+}
+
+static const struct {
+ void (*func)(const u8_t *data);
+ u16_t len;
+} prov_handlers[] = {
+ { prov_invite, 1 },
+ { prov_capabilities, 11 },
+ { prov_start, 5 },
+ { prov_pub_key, 64 },
+ { prov_input_complete, 0 },
+ { prov_confirm, 16 },
+ { prov_random, 16 },
+ { prov_data, 33 },
+ { prov_complete, 0 },
+ { prov_failed, 1 },
+};
+
+static void close_link(int i, u8_t reason)
+{
+ if (i < CONFIG_BT_MESH_PBA_SAME_TIME) {
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ bearer_ctl_send(i, LINK_CLOSE, &reason, sizeof(reason));
+#endif
+ } else if (i >= CONFIG_BT_MESH_PBA_SAME_TIME &&
+ i < BT_MESH_PROV_SAME_TIME) {
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (link[i].conn) {
+ bt_mesh_gattc_disconnect(link[i].conn);
+ }
+#endif
+ } else {
+ BT_ERR("%s, Invalid link index %d", __func__, i);
+ }
+}
+
+static void prov_timeout(struct k_work *work)
+{
+ int i = work->index;
+
+ BT_DBG("%s", __func__);
+
+ close_link(i, CLOSE_REASON_TIMEOUT);
+}
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+static void prov_retransmit(struct k_work *work)
+{
+ int id = work->index;
+
+ BT_DBG("%s", __func__);
+
+ if (!atomic_test_bit(link[id].flags, LINK_ACTIVE)) {
+ BT_WARN("%s, Link is not active", __func__);
+ return;
+ }
+
+ if (k_uptime_get() - link[id].tx.start > TRANSACTION_TIMEOUT) {
+ BT_WARN("%s, Timeout, giving up transaction", __func__);
+ close_link(id, CLOSE_REASON_TIMEOUT);
+ return;
+ }
+
+ if (link[id].link_close & BIT(0)) {
+ if (link[id].link_close >> 1 & 0x02) {
+ reset_link(id, link[id].link_close >> 8);
+ return;
+ }
+ link[id].link_close += BIT(1);
+ }
+
+ for (int i = 0; i < ARRAY_SIZE(link[id].tx.buf); i++) {
+ struct net_buf *buf = link[id].tx.buf[i];
+
+ if (!buf) {
+ break;
+ }
+
+ if (BT_MESH_ADV(buf)->busy) {
+ continue;
+ }
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (i + 1 < ARRAY_SIZE(link[id].tx.buf) && link[id].tx.buf[i + 1]) {
+ bt_mesh_adv_send(buf, NULL, NULL);
+ } else {
+ bt_mesh_adv_send(buf, &buf_sent_cb, (void *)id);
+ }
+ }
+}
+
+static void link_ack(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ int i = prov_get_pb_index();
+
+ BT_DBG("len %u", buf->len);
+
+ if (buf->len) {
+ BT_ERR("%s, Invalid Link ACK length", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (link[i].expect == PROV_CAPABILITIES) {
+ BT_WARN("%s, Link ACK is already received", __func__);
+ return;
+ }
+
+ link[i].conf_inputs = (u8_t *)osi_calloc(PROV_CONF_INPUTS_SIZE);
+ if (!link[i].conf_inputs) {
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ /** After received link_ack, we don't call prov_clear_tx() to
+ * cancel retransmit timer, because retransmit timer will be
+ * cancelled after we send the provisioning invite pdu.
+ */
+ send_invite();
+}
+
+static void link_close(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ u8_t reason;
+ int i = prov_get_pb_index();
+
+ BT_DBG("len %u", buf->len);
+
+ reason = net_buf_simple_pull_u8(buf);
+
+ reset_link(i, reason);
+}
+
+static void gen_prov_ctl(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ int i = prov_get_pb_index();
+
+ BT_DBG("op 0x%02x len %u", BEARER_CTL(rx->gpc), buf->len);
+
+ switch (BEARER_CTL(rx->gpc)) {
+ case LINK_OPEN:
+ break;
+
+ case LINK_ACK:
+ if (!atomic_test_bit(link[i].flags, LINK_ACTIVE)) {
+ return;
+ }
+ link_ack(rx, buf);
+ break;
+
+ case LINK_CLOSE:
+ if (!atomic_test_bit(link[i].flags, LINK_ACTIVE)) {
+ return;
+ }
+ link_close(rx, buf);
+ break;
+
+ default:
+ BT_ERR("%s, Unknown bearer opcode 0x%02x", __func__, BEARER_CTL(rx->gpc));
+ return;
+ }
+}
+
+static void prov_msg_recv(void)
+{
+ int i = prov_get_pb_index();
+
+ u8_t type = link[i].rx.buf->data[0];
+
+ BT_DBG("type 0x%02x len %u", type, link[i].rx.buf->len);
+
+ /** Provisioner first checks information of the received
+ * provisioing pdu, and once succeed, check the fcs
+ */
+ if (type != PROV_FAILED && type != link[i].expect) {
+ BT_ERR("%s, Unexpected msg 0x%02x != 0x%02x", __func__, type, link[i].expect);
+ goto fail;
+ }
+
+ if (type >= 0x0A) {
+ BT_ERR("%s, Unknown provisioning PDU type 0x%02x", __func__, type);
+ goto fail;
+ }
+
+ if (1 + prov_handlers[type].len != link[i].rx.buf->len) {
+ BT_ERR("%s, Invalid length %u for type 0x%02x", __func__, link[i].rx.buf->len, type);
+ goto fail;
+ }
+
+ if (!bt_mesh_fcs_check(link[i].rx.buf, link[i].rx.fcs)) {
+ BT_ERR("%s, Incorrect FCS", __func__);
+ goto fail;
+ }
+
+ gen_prov_ack_send(link[i].rx.trans_id);
+ link[i].rx.prev_id = link[i].rx.trans_id;
+ link[i].rx.trans_id = 0;
+
+ prov_handlers[type].func(&link[i].rx.buf->data[1]);
+ return;
+
+fail:
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+}
+
+static void gen_prov_cont(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ u8_t seg = CONT_SEG_INDEX(rx->gpc);
+ int i = prov_get_pb_index();
+
+ BT_DBG("len %u, seg_index %u", buf->len, seg);
+
+ if (!link[i].rx.seg && link[i].rx.prev_id == rx->xact_id) {
+ BT_WARN("%s, Resending ack", __func__);
+ gen_prov_ack_send(rx->xact_id);
+ return;
+ }
+
+ if (rx->xact_id != link[i].rx.trans_id) {
+ BT_WARN("%s, Data for unknown transaction (%u != %u)",
+ __func__, rx->xact_id, link[i].rx.trans_id);
+ /* If Provisioner receives a Provisioning PDU with a mismatch
+ * transaction number, it just ignore the PDU.
+ */
+ return;
+ }
+
+ if (seg > link[i].rx.last_seg) {
+ BT_ERR("%s, Invalid segment index %u", __func__, seg);
+ goto fail;
+ } else if (seg == link[i].rx.last_seg) {
+ u8_t expect_len;
+
+ expect_len = (link[i].rx.buf->len - 20 -
+ (23 * (link[i].rx.last_seg - 1)));
+ if (expect_len != buf->len) {
+ BT_ERR("%s, Incorrect last seg len: %u != %u",
+ __func__, expect_len, buf->len);
+ goto fail;
+ }
+ }
+
+ if (!(link[i].rx.seg & BIT(seg))) {
+ BT_WARN("%s, Ignore already received segment", __func__);
+ return;
+ }
+
+ memcpy(XACT_SEG_DATA(seg), buf->data, buf->len);
+ XACT_SEG_RECV(seg);
+
+ if (!link[i].rx.seg) {
+ prov_msg_recv();
+ }
+ return;
+
+fail:
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+}
+
+static void gen_prov_ack(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ int i = prov_get_pb_index();
+ u8_t ack_type, pub_key_oob;
+
+ BT_DBG("len %u", buf->len);
+
+ if (!link[i].tx.buf[0]) {
+ return;
+ }
+
+ if (!link[i].tx.trans_id) {
+ return;
+ }
+
+ if (rx->xact_id == (link[i].tx.trans_id - 1)) {
+ prov_clear_tx(i);
+
+ ack_type = link[i].expect_ack_for;
+ switch (ack_type) {
+ case PROV_START:
+ pub_key_oob = link[i].conf_inputs[13];
+ send_pub_key(pub_key_oob);
+ break;
+ case PROV_PUB_KEY:
+ prov_gen_dh_key(i);
+ break;
+ default:
+ break;
+ }
+ link[i].expect_ack_for = 0x00;
+ }
+}
+
+static void gen_prov_start(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ int i = prov_get_pb_index();
+
+ if (link[i].rx.seg) {
+ BT_WARN("%s, Get Start while there are unreceived segments", __func__);
+ return;
+ }
+
+ if (link[i].rx.prev_id == rx->xact_id) {
+ BT_WARN("%s, Resending ack", __func__);
+ gen_prov_ack_send(rx->xact_id);
+ return;
+ }
+
+ link[i].rx.buf->len = net_buf_simple_pull_be16(buf);
+ link[i].rx.trans_id = rx->xact_id;
+ link[i].rx.fcs = net_buf_simple_pull_u8(buf);
+
+ BT_DBG("len %u last_seg %u total_len %u fcs 0x%02x", buf->len,
+ START_LAST_SEG(rx->gpc), link[i].rx.buf->len, link[i].rx.fcs);
+
+ /* Provisioner can not receive zero-length provisioning pdu */
+ if (link[i].rx.buf->len < 1) {
+ BT_ERR("%s, Ignoring zero-length provisioning PDU", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (link[i].rx.buf->len > link[i].rx.buf->size) {
+ BT_ERR("%s, Too large provisioning PDU (%u bytes)", __func__, link[i].rx.buf->len);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ if (START_LAST_SEG(rx->gpc) > 0 && link[i].rx.buf->len <= 20) {
+ BT_ERR("%s, Too small total length for multi-segment PDU", __func__);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ link[i].rx.seg = (1 << (START_LAST_SEG(rx->gpc) + 1)) - 1;
+ link[i].rx.last_seg = START_LAST_SEG(rx->gpc);
+ memcpy(link[i].rx.buf->data, buf->data, buf->len);
+ XACT_SEG_RECV(0);
+
+ if (!link[i].rx.seg) {
+ prov_msg_recv();
+ }
+}
+
+static const struct {
+ void (*const func)(struct prov_rx *rx, struct net_buf_simple *buf);
+ const u8_t require_link;
+ const u8_t min_len;
+} gen_prov[] = {
+ { gen_prov_start, true, 3 },
+ { gen_prov_ack, true, 0 },
+ { gen_prov_cont, true, 0 },
+ { gen_prov_ctl, true, 0 },
+};
+
+static void gen_prov_recv(struct prov_rx *rx, struct net_buf_simple *buf)
+{
+ int i = prov_get_pb_index();
+
+ if (buf->len < gen_prov[GPCF(rx->gpc)].min_len) {
+ BT_ERR("%s, Too short GPC message type %u", __func__, GPCF(rx->gpc));
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ /** require_link flag can be used combined with link[].linking flag
+ * to set LINK_ACTIVE status after link_ack pdu is received.
+ * And if so, we shall not check LINK_ACTIVE status in the
+ * function find_link().
+ */
+ if (!atomic_test_bit(link[i].flags, LINK_ACTIVE) &&
+ gen_prov[GPCF(rx->gpc)].require_link) {
+ BT_DBG("Ignoring message that requires active link");
+ return;
+ }
+
+ gen_prov[GPCF(rx->gpc)].func(rx, buf);
+}
+
+static int find_link(u32_t link_id, bool set)
+{
+ int i;
+
+ /* link for PB-ADV is from 0 to CONFIG_BT_MESH_PBA_SAME_TIME */
+ for (i = 0; i < CONFIG_BT_MESH_PBA_SAME_TIME; i++) {
+ if (atomic_test_bit(link[i].flags, LINK_ACTIVE)) {
+ if (link[i].link_id == link_id) {
+ if (set) {
+ prov_set_pb_index(i);
+ }
+ return 0;
+ }
+ }
+ }
+
+ return -1;
+}
+
+void provisioner_pb_adv_recv(struct net_buf_simple *buf)
+{
+ struct prov_rx rx;
+ int i;
+
+ rx.link_id = net_buf_simple_pull_be32(buf);
+ if (find_link(rx.link_id, true) < 0) {
+ BT_DBG("%s, Data for unexpected link", __func__);
+ return;
+ }
+
+ i = prov_get_pb_index();
+
+ if (buf->len < 2) {
+ BT_ERR("%s, Too short provisioning packet (len %u)", __func__, buf->len);
+ close_link(i, CLOSE_REASON_FAILED);
+ return;
+ }
+
+ rx.xact_id = net_buf_simple_pull_u8(buf);
+ rx.gpc = net_buf_simple_pull_u8(buf);
+
+ BT_DBG("link_id 0x%08x xact_id %u", rx.link_id, rx.xact_id);
+
+ gen_prov_recv(&rx, buf);
+}
+#endif /* CONFIG_BT_MESH_PB_ADV */
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+static struct bt_conn *find_conn(struct bt_conn *conn, bool set)
+{
+ int i;
+
+ /* link for PB-GATT is from CONFIG_BT_MESH_PBA_SAME_TIME to BT_MESH_PROV_SAME_TIME */
+ for (i = CONFIG_BT_MESH_PBA_SAME_TIME; i < BT_MESH_PROV_SAME_TIME; i++) {
+ if (atomic_test_bit(link[i].flags, LINK_ACTIVE)) {
+ if (link[i].conn == conn) {
+ if (set) {
+ prov_set_pb_index(i);
+ }
+ return conn;
+ }
+ }
+ }
+
+ return NULL;
+}
+
+int provisioner_pb_gatt_recv(struct bt_conn *conn, struct net_buf_simple *buf)
+{
+ u8_t type;
+ int i;
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (!find_conn(conn, true)) {
+ BT_ERR("%s, Data for unexpected connection", __func__);
+ return -ENOTCONN;
+ }
+
+ i = prov_get_pb_index();
+
+ if (buf->len < 1) {
+ BT_ERR("%s, Too short provisioning packet (len %u)", __func__, buf->len);
+ goto fail;
+ }
+
+ type = net_buf_simple_pull_u8(buf);
+ if (type != PROV_FAILED && type != link[i].expect) {
+ BT_ERR("%s, Unexpected msg 0x%02x != 0x%02x", __func__, type, link[i].expect);
+ goto fail;
+ }
+
+ if (type >= 0x0A) {
+ BT_ERR("%s, Unknown provisioning PDU type 0x%02x", __func__, type);
+ goto fail;
+ }
+
+ if (prov_handlers[type].len != buf->len) {
+ BT_ERR("%s, Invalid length %u for type 0x%02x", __func__, buf->len, type);
+ goto fail;
+ }
+
+ prov_handlers[type].func(buf->data);
+
+ return 0;
+
+fail:
+ /* Mesh Spec Section 5.4.4 Provisioning errors */
+ close_link(i, CLOSE_REASON_FAILED);
+ return -EINVAL;
+}
+
+int provisioner_set_prov_conn(const u8_t addr[6], struct bt_conn *conn)
+{
+ if (!addr || !conn) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ for (int i = CONFIG_BT_MESH_PBA_SAME_TIME; i < BT_MESH_PROV_SAME_TIME; i++) {
+ if (!memcmp(link[i].addr.a.val, addr, BD_ADDR_LEN)) {
+ link[i].conn = bt_mesh_conn_ref(conn);
+ return 0;
+ }
+ }
+
+ BT_ERR("%s, Address %s is not found", __func__, bt_hex(addr, BD_ADDR_LEN));
+ return -ENOMEM;
+}
+
+int provisioner_pb_gatt_open(struct bt_conn *conn, u8_t *addr)
+{
+ int i, id = 0;
+
+ BT_DBG("conn %p", conn);
+
+ /** Double check if the device is currently being provisioned
+ * using PB-ADV.
+ * Provisioner binds conn with proper device when
+ * proxy_prov_connected() is invoked, and here after proper GATT
+ * procedures are completed, we just check if this conn already
+ * exists in the proxy servers array.
+ */
+ for (i = CONFIG_BT_MESH_PBA_SAME_TIME; i < BT_MESH_PROV_SAME_TIME; i++) {
+ if (link[i].conn == conn) {
+ id = i;
+ break;
+ }
+ }
+
+ if (i == BT_MESH_PROV_SAME_TIME) {
+ BT_ERR("%s, Link is not found", __func__);
+ return -ENOTCONN;
+ }
+
+ prov_set_pb_index(id);
+
+ for (i = 0; i < CONFIG_BT_MESH_PBA_SAME_TIME; i++) {
+ if (atomic_test_bit(link[i].flags, LINK_ACTIVE)) {
+ if (!memcmp(link[i].uuid, link[id].uuid, 16)) {
+ BT_WARN("%s, Provision using PB-GATT & PB-ADV same time", __func__);
+ close_link(id, CLOSE_REASON_FAILED);
+ return -EALREADY;
+ }
+ }
+ }
+
+ atomic_set_bit(link[id].flags, LINK_ACTIVE);
+ link[id].conn = bt_mesh_conn_ref(conn);
+
+ /* May use lcd to indicate starting provisioning each device */
+ if (prov->prov_link_open) {
+ prov->prov_link_open(BT_MESH_PROV_GATT);
+ }
+
+ link[id].conf_inputs = (u8_t *)osi_calloc(PROV_CONF_INPUTS_SIZE);
+ if (!link[id].conf_inputs) {
+ /* Disconnect this connection, clear corresponding informations */
+ BT_ERR("%s, Failed to allocate memory", __func__);
+ close_link(id, CLOSE_REASON_FAILED);
+ return -ENOMEM;
+ }
+
+ send_invite();
+
+ return 0;
+}
+
+int provisioner_pb_gatt_close(struct bt_conn *conn, u8_t reason)
+{
+ int i;
+
+ BT_DBG("conn %p", conn);
+
+ if (!find_conn(conn, true)) {
+ BT_ERR("%s, Conn %p is not found", __func__, conn);
+ return -ENOTCONN;
+ }
+
+ i = prov_get_pb_index();
+
+ if (atomic_test_and_clear_bit(link[i].flags, TIMEOUT_START)) {
+ k_delayed_work_cancel(&link[i].timeout);
+ }
+
+ if (prov->prov_link_close) {
+ prov->prov_link_close(BT_MESH_PROV_GATT, reason);
+ }
+
+ prov_memory_free(i);
+
+ memset(&link[i], 0, offsetof(struct prov_link, timeout));
+
+ if (bt_mesh_pub_key_get()) {
+ atomic_set_bit(link[i].flags, LOCAL_PUB_KEY);
+ }
+
+ return 0;
+}
+#endif /* CONFIG_BT_MESH_PB_GATT */
+
+int provisioner_prov_init(const struct bt_mesh_prov *prov_info)
+{
+ const u8_t *key = NULL;
+ int i;
+
+ if (!prov_info) {
+ BT_ERR("%s, No provisioning context provided", __func__);
+ return -EINVAL;
+ }
+
+ if (CONFIG_BT_MESH_PBG_SAME_TIME > CONFIG_BT_MAX_CONN) {
+ BT_ERR("%s, PB-GATT same time exceeds max connection", __func__);
+ return -EINVAL;
+ }
+
+ key = bt_mesh_pub_key_get();
+ if (!key) {
+ BT_ERR("%s, Failed to generate Public Key", __func__);
+ return -EIO;
+ }
+
+ prov = prov_info;
+
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ for (i = 0; i < CONFIG_BT_MESH_PBA_SAME_TIME; i++) {
+ adv_buf[i].buf.size = ADV_BUF_SIZE;
+ link[i].pending_ack = XACT_NVAL;
+ k_delayed_work_init(&link[i].tx.retransmit, prov_retransmit);
+ link[i].tx.retransmit.work.index = i;
+ link[i].rx.prev_id = XACT_NVAL;
+ link[i].rx.buf = bt_mesh_pba_get_buf(i);
+ }
+#endif
+
+ for (i = 0; i < BT_MESH_PROV_SAME_TIME; i++) {
+ k_delayed_work_init(&link[i].timeout, prov_timeout);
+ link[i].timeout.work.index = i;
+ }
+
+ /* for PB-GATT, use servers[] array in proxy_provisioner.c */
+
+ prov_ctx.current_addr = prov->prov_start_address;
+ prov_ctx.curr_net_idx = BT_MESH_KEY_PRIMARY;
+ prov_ctx.curr_flags = prov->flags;
+ prov_ctx.curr_iv_index = prov->iv_index;
+
+ return 0;
+}
+
+static int provisioner_notify_unprov_adv_pkt(bt_mesh_prov_bearer_t bearer, const u8_t uuid[16],
+ const bt_addr_le_t *addr, u16_t oob_info)
+{
+ u8_t adv_type;
+ int index;
+
+ if ((bearer != BT_MESH_PROV_ADV && bearer != BT_MESH_PROV_GATT) || !uuid || !addr) {
+ BT_ERR("%s, invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ adv_type = (bearer == BT_MESH_PROV_ADV) ? BT_LE_ADV_NONCONN_IND : BT_LE_ADV_IND;
+
+ if (prov_ctx.prov_after_match == false) {
+ if (provisioner_dev_find(addr, uuid, &index)) {
+ BT_DBG("%s, Device is not found, notify to upper layer", __func__);
+ if (notify_unprov_adv_pkt_cb) {
+ notify_unprov_adv_pkt_cb(addr->a.val, addr->type, adv_type, uuid, oob_info, bearer);
+ }
+ return -EALREADY;
+ }
+
+ if (!(unprov_dev[index].bearer & bearer)) {
+ BT_DBG("%s, Not support PB-%s", __func__, (bearer == BT_MESH_PROV_ADV) ? "ADV" : "GATT");
+ return -EIO;
+ }
+ }
+
+ return 0;
+}
+
+void provisioner_unprov_beacon_recv(struct net_buf_simple *buf)
+{
+#if defined(CONFIG_BT_MESH_PB_ADV)
+ const bt_addr_le_t *addr = NULL;
+ u8_t *uuid = NULL;
+ u16_t oob_info;
+
+ if (buf->len != 0x12 && buf->len != 0x16) {
+ BT_ERR("%s, Invalid Unprovisioned Device Beacon length", __func__);
+ return;
+ }
+
+ if (prov_ctx.pba_count == CONFIG_BT_MESH_PBA_SAME_TIME) {
+ BT_DBG("Current PB-ADV devices reach max limit");
+ return;
+ }
+
+ addr = bt_mesh_pba_get_addr();
+ uuid = buf->data;
+ net_buf_simple_pull(buf, 16);
+ /* Mesh beacon uses big-endian to send beacon data */
+ oob_info = net_buf_simple_pull_be16(buf);
+
+ if (provisioner_check_device_uuid(uuid)) {
+ return;
+ }
+
+ if (provisioner_notify_unprov_adv_pkt(
+ BT_MESH_PROV_ADV, uuid, addr, oob_info)) {
+ return;
+ }
+
+ provisioner_start_prov_device(BT_MESH_PROV_ADV, uuid, addr, oob_info);
+#endif /* CONFIG_BT_MESH_PB_ADV */
+}
+
+bool provisioner_flags_match(struct net_buf_simple *buf)
+{
+ u8_t flags;
+
+ if (buf->len != 1) {
+ BT_DBG("%s, Unexpected flags length", __func__);
+ return false;
+ }
+
+ flags = net_buf_simple_pull_u8(buf);
+
+ BT_DBG("Received adv pkt with flags: 0x%02x", flags);
+
+ /* Flags context will not be checked curently */
+
+ return true;
+}
+
+u16_t provisioner_srv_uuid_recv(struct net_buf_simple *buf)
+{
+ u16_t uuid = 0;
+
+ if (buf->len != 2) {
+ BT_DBG("Length not match mesh service uuid");
+ return false;
+ }
+
+ uuid = net_buf_simple_pull_le16(buf);
+
+ BT_DBG("Received adv pkt with service UUID: %d", uuid);
+
+ if ((uuid != BT_UUID_MESH_PROV_VAL) && (uuid != BT_UUID_MESH_PROXY_VAL)) {
+ return false;
+ }
+
+ return uuid;
+}
+
+static void provisioner_prov_srv_data_recv(struct net_buf_simple *buf, const bt_addr_le_t *addr);
+
+void provisioner_srv_data_recv(struct net_buf_simple *buf, const bt_addr_le_t *addr, u16_t uuid)
+{
+ u16_t uuid_type;
+
+ if (!buf || !addr) {
+ BT_ERR("%s, Invalid parameter", __func__);
+ return;
+ }
+
+ uuid_type = net_buf_simple_pull_le16(buf);
+ if (uuid_type != uuid) {
+ BT_DBG("%s, Invalid Mesh Service Data UUID 0x%04x", __func__, uuid_type);
+ return;
+ }
+
+ switch (uuid) {
+ case BT_UUID_MESH_PROV_VAL:
+ if (buf->len != BT_MESH_PROV_SRV_DATA_LEN) {
+ BT_WARN("%s, Invalid Mesh Prov Service Data length %d", __func__, buf->len);
+ return;
+ }
+ BT_DBG("Start to deal with Mesh Prov Service Data");
+ provisioner_prov_srv_data_recv(buf, addr);
+ break;
+ case BT_UUID_MESH_PROXY_VAL:
+ if (buf->len != BT_MESH_PROXY_SRV_DATA_LEN1 &&
+ buf->len != BT_MESH_PROXY_SRV_DATA_LEN2) {
+ BT_ERR("%s, Invalid Mesh Proxy Service Data length %d", __func__, buf->len);
+ return;
+ }
+ BT_DBG("Start to deal with Mesh Proxy Service Data");
+ provisioner_proxy_srv_data_recv(buf);
+ break;
+ default:
+ break;
+ }
+}
+
+static void provisioner_prov_srv_data_recv(struct net_buf_simple *buf, const bt_addr_le_t *addr)
+{
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ u8_t *uuid = NULL;
+ u16_t oob_info;
+
+ if (prov_ctx.pbg_count == CONFIG_BT_MESH_PBG_SAME_TIME) {
+ BT_DBG("Current PB-GATT devices reach max limit");
+ return;
+ }
+
+ uuid = buf->data;
+ net_buf_simple_pull(buf, 16);
+ /* Mesh beacon uses big-endian to send beacon data */
+ oob_info = net_buf_simple_pull_be16(buf);
+
+ if (provisioner_check_device_uuid(uuid)) {
+ return;
+ }
+
+ if (provisioner_notify_unprov_adv_pkt(
+ BT_MESH_PROV_GATT, uuid, addr, oob_info)) {
+ return;
+ }
+
+ /* Provisioner will copy the device uuid, oob info, etc. into an unused link
+ * struct, and at this moment the link has not been activated. Even if we
+ * receive an Unprovisioned Device Beacon and a Connectable Provisioning adv
+ * pkt from the same device, and store the device info received within each
+ * adv pkt into two link structs which will has no impact on the provisioning
+ * of this device, because no matter which link among PB-GATT and PB-ADV is
+ * activated first, the other one will be dropped finally and the link struct
+ * occupied by the dropped link will be used by other devices (because the link
+ * is not activated).
+ * Use connecting flag to prevent if two devices's adv pkts are both received,
+ * the previous one info will be replaced by the second one.
+ */
+ provisioner_start_prov_device(BT_MESH_PROV_GATT, uuid, addr, oob_info);
+#endif /* CONFIG_BT_MESH_PB_GATT */
+}
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+#endif /* CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_prov.h b/components/bt/ble_mesh/mesh_core/provisioner_prov.h
new file mode 100644
index 0000000000..d0867b059d
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_prov.h
@@ -0,0 +1,381 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _PROVISIONER_PROV_H_
+#define _PROVISIONER_PROV_H_
+
+#include "mesh_bearer_adapt.h"
+#include "mesh_main.h"
+
+#if !CONFIG_BT_MESH_PROVISIONER
+
+#define CONFIG_BT_MESH_PBA_SAME_TIME 0
+#define CONFIG_BT_MESH_PBG_SAME_TIME 0
+
+#else
+
+#if !defined(CONFIG_BT_MESH_PB_ADV)
+#define CONFIG_BT_MESH_PBA_SAME_TIME 0
+#endif /* !CONFIG_BT_MESH_PB_ADV */
+
+#if !defined(CONFIG_BT_MESH_PB_GATT)
+#define CONFIG_BT_MESH_PBG_SAME_TIME 0
+#endif /* !CONFIG_BT_MESH_PB_GATT */
+
+#define BT_DATA_FLAGS 0x01
+#define BT_DATA_SERVICE_UUID 0x03
+#define BT_DATA_SERVICE_DATA 0X16
+
+#define RM_AFTER_PROV BIT(0)
+#define START_PROV_NOW BIT(1)
+#define FLUSHABLE_DEV BIT(2)
+
+struct bt_mesh_unprov_dev_add {
+ u8_t addr[6];
+ u8_t addr_type;
+ u8_t uuid[16];
+ u16_t oob_info;
+ u8_t bearer;
+};
+
+struct bt_mesh_device_delete {
+ u8_t addr[6];
+ u8_t addr_type;
+ u8_t uuid[16];
+};
+
+#define NET_IDX_FLAG BIT(0)
+#define FLAGS_FLAG BIT(1)
+#define IV_INDEX_FLAG BIT(2)
+
+struct bt_mesh_prov_data_info {
+ union {
+ u16_t net_idx;
+ u8_t flags;
+ u32_t iv_index;
+ };
+ u8_t flag;
+};
+
+/* The following APIs are for primary provisioner internal use */
+
+/**
+ * @brief This function decrements the current PB-GATT count.
+ *
+ * @return None
+ */
+void provisioner_pbg_count_dec(void);
+
+/**
+ * @brief This function increments the current PB-GATT count.
+ *
+ * @return None
+ */
+void provisioner_pbg_count_inc(void);
+
+/**
+ * @brief This function clears the part of the link info of the proper device.
+ *
+ * @param[in] addr: Remote device address
+ *
+ * @return None
+ */
+void provisioner_clear_link_conn_info(u8_t addr[6]);
+
+/**
+ * @brief This function handles the received PB-ADV PDUs.
+ *
+ * @param[in] buf: Pointer to the buffer containing generic provisioning PDUs
+ *
+ * @return Zero - success, otherwise - fail
+ */
+void provisioner_pb_adv_recv(struct net_buf_simple *buf);
+
+/**
+ * @brief This function sends provisioning invite to start
+ * provisioning this unprovisioned device.
+ *
+ * @param[in] addr: Remote device address
+ * @param[in] conn: Pointer to the bt_conn structure
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int provisioner_set_prov_conn(const u8_t addr[6], struct bt_conn *conn);
+
+/**
+ * @brief This function sends provisioning invite to start
+ * provisioning this unprovisioned device.
+ *
+ * @param[in] conn: Pointer to the bt_conn structure
+ * @param[in] addr: Address of the connected device
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int provisioner_pb_gatt_open(struct bt_conn *conn, u8_t *addr);
+
+/**
+ * @brief This function resets the used information when
+ * related connection is terminated.
+ *
+ * @param[in] conn: Pointer to the bt_conn structure
+ * @param[in] reason: Connection terminated reason
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int provisioner_pb_gatt_close(struct bt_conn *conn, u8_t reason);
+
+/**
+ * @brief This function handles the received PB-GATT provision
+ * PDUs.
+ *
+ * @param[in] conn: Pointer to the bt_conn structure
+ * @param[in] buf: Pointer to the buffer containing provision PDUs
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int provisioner_pb_gatt_recv(struct bt_conn *conn, struct net_buf_simple *buf);
+
+/**
+ * @brief This function initializes provisioner's PB-GATT and PB-ADV
+ * related information.
+ *
+ * @param[in] prov_info: Pointer to the application-initialized provisioner info.
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int provisioner_prov_init(const struct bt_mesh_prov *prov_info);
+
+/**
+ * @brief This function parses the received unprovisioned device
+ * beacon advertising packets, and if checked, starts to provision this device
+ * using PB-ADV bearer.
+ *
+ * @param[in] buf: Pointer to the buffer containing unprovisioned device beacon
+ *
+ * @return None
+ */
+void provisioner_unprov_beacon_recv(struct net_buf_simple *buf);
+
+/**
+ * @brief This function parses the flags part of the
+ * received connectable mesh provisioning advertising packets.
+ *
+ * @param[in] buf: Pointer to the buffer containing advertising flags part
+ *
+ * @return True - success, False - fail
+ */
+bool provisioner_flags_match(struct net_buf_simple *buf);
+
+/**
+ * @brief This function parses the service UUID part of the
+ * received connectable mesh provisioning advertising packets.
+ *
+ * @param[in] buf: Pointer to the buffer containing service UUID part
+ *
+ * @return Zero - fail, otherwise - Service UUID(0x1827 or 0x1828)
+ */
+u16_t provisioner_srv_uuid_recv(struct net_buf_simple *buf);
+
+/**
+ * @brief This function parses the service data part of the
+ * received connectable mesh provisioning advertising packets.
+ *
+ * @param[in] buf: Pointer to the buffer containing the remianing service data part
+ * @param[in] addr: Pointer to the received device address
+ * @param[in] uuid: Service UUID contained in the service UUID part
+ *
+ * @return None
+ */
+void provisioner_srv_data_recv(struct net_buf_simple *buf, const bt_addr_le_t *addr, u16_t uuid);
+
+/**
+ * @brief This function gets the bt_mesh_prov pointer.
+ *
+ * @return bt_mesh_prov pointer(prov)
+ */
+const struct bt_mesh_prov *provisioner_get_prov_info(void);
+
+/**
+ * @brief This function resets all nodes information in provisioner_prov.c.
+ *
+ * @return Zero
+ */
+int provisioner_prov_reset_all_nodes(void);
+
+/* The following APIs are for primary provisioner application use */
+
+/** @brief Add unprovisioned device info to unprov_dev queue
+ *
+ * @param[in] add_dev: Pointer to the structure containing the device information
+ * @param[in] flags: Flags indicate several operations of the device information
+ * - Remove device information from queue after it is provisioned (BIT0)
+ * - Start provisioning as soon as device is added to queue (BIT1)
+ * - Device can be flushed when device queue is full (BIT2)
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ *
+ * @Note: 1. Currently address type only supports public address and static random address.
+ * 2. If device UUID and/or device address and address type already exist in the
+ * device queue, but the bearer differs from the existing one, add operation
+ * will also be successful and it will update the provision bearer supported by
+ * the device.
+ */
+int bt_mesh_provisioner_add_unprov_dev(struct bt_mesh_unprov_dev_add *add_dev, u8_t flags);
+
+/** @brief Delete device from queue, reset current provisioning link and reset the node
+ *
+ * @param[in] del_dev: Pointer to the structure containing the device information
+ *
+ * @return Zero on success or (negative) error code otherwise.
+ */
+int bt_mesh_provisioner_delete_device(struct bt_mesh_device_delete *del_dev);
+
+/**
+ * @brief This function sets a part of the device UUID for comparison before
+ * starting to provision the device.
+ *
+ * @param[in] offset: offset of the device UUID to be compared
+ * @param[in] length: length of the device UUID to be compared
+ * @param[in] match: value to be compared
+ * @param[in] prov_flag: flags indicate if uuid_match advertising packets are received, after that
+ * the device will be provisioned at once or reported to the application layer
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int bt_mesh_provisioner_set_dev_uuid_match(u8_t offset, u8_t length,
+ const u8_t *match, bool prov_flag);
+
+/** @brief Callback for provisioner receiving advertising packet from unprovisioned devices which are
+ * not in the unprovisioned device queue.
+ *
+ * Report on the unprovisioned device beacon and mesh provisioning service advertising data to application layer
+ *
+ * @param addr Unprovisioned device address pointer
+ * @param addr_type Unprovisioned device address type
+ * @param dev_uuid Unprovisioned device device UUID pointer
+ * @param bearer Advertising packet received from PB-GATT or PB-ADV bearer
+ * @param adv_type Adv packet type, currently this is not used and we can use bearer to device
+ * the adv_type(ADV_IND or ADV_NONCONN_IND). This parameter will be used, when
+ * scan response data will be supported.
+ *
+ */
+typedef void (*unprov_adv_pkt_cb_t)(const u8_t addr[6], const u8_t addr_type,
+ const u8_t adv_type, const u8_t dev_uuid[16],
+ u16_t oob_info, bt_mesh_prov_bearer_t bearer);
+
+/**
+ * @brief This function registers the callback which notifies the application
+ * layer of the received mesh provisioning or unprovisioned device
+ * beacon advertizing packets (from devices not in the unprov device queue).
+ *
+ * @param[in] cb: Callback of the notifying adv pkts function
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int bt_mesh_prov_adv_pkt_cb_register(unprov_adv_pkt_cb_t cb);
+
+/**
+ * @brief This function changes net_idx or flags or iv_index used in provisioning data.
+ *
+ * @param[in] info: Pointer of structure containing net_idx or flags or iv_index
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int bt_mesh_provisioner_set_prov_data_info(struct bt_mesh_prov_data_info *info);
+
+/**
+ * @brief This function is called to input number/string out-put by unprovisioned device.
+ *
+ * @param[in] val Pointer of the input number/string
+ * @param[in] link_idx The provisioning link index
+ * @param[in] num_flag Flag indicates if it is a number or string
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int bt_mesh_prov_set_oob_input_data(u8_t *val, u8_t link_idx, bool num_flag);
+
+/**
+ * @brief This function is called to output number/string which will be input by unprovisioned device.
+ *
+ * @param[in] num Pointer of the output number/string
+ * @param[in] size Size of the output number/string
+ * @param[in] num_flag Flag indicates if it is a number or string
+ * @param[in] link_idx The provisioning link index
+ *
+ * @return Zero - success, otherwise - fail
+ */
+// int bt_mesh_prov_set_oob_output_data(u8_t *num, u8_t size, bool num_flag, u8_t link_idx);
+
+/**
+ * @brief This function is called to read unprovisioned device's oob public key.
+ *
+ * @param[in] link_idx The provisioning link index
+ * @param[in] pub_key_x Unprovisioned device's Public Key X
+ * @param[in] pub_key_y Unprovisioned device's Public Key Y
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int bt_mesh_prov_read_oob_pub_key(u8_t link_idx, u8_t pub_key_x[32], u8_t pub_key_y[32]);
+
+/* The following APIs are for fast provisioning */
+
+/**
+ * @brief This function is called to set fast_prov_flag.
+ *
+ * @param[in] flag: Flag set to fast_prov_flag
+ *
+ * @return None
+ */
+void provisioner_set_fast_prov_flag(bool flag);
+
+/**
+ * @brief This function is called to set netkey index used for fast provisioning.
+ *
+ * @param[in] net_key: Netkey value
+ * @param[in] net_idx: Netkey index
+ *
+ * @return status for set netkey index msg
+ */
+u8_t provisioner_set_fast_prov_net_idx(const u8_t *net_key, u16_t net_idx);
+
+/**
+ * @brief This function is called to get netkey index used for fast provisioning.
+ *
+ * @return net_idx of fast provisioning
+ */
+u16_t provisioner_get_fast_prov_net_idx(void);
+
+/**
+ * @brief This function is called to set unicast address range used for fast provisioning.
+ *
+ * @param[in] min: Minimum unicast address
+ * @param[in] max: Maximum unicast address
+ *
+ * @return status for set unicast address range message
+ */
+u8_t bt_mesh_set_fast_prov_unicast_addr_range(u16_t min, u16_t max);
+
+/**
+ * @brief This function is called to set flags & iv_index used for fast provisioning.
+ *
+ * @param[in] flags: Key refresh flag and iv update flag
+ * @param[in] iv_index: IV index
+ *
+ * @return None
+ */
+void bt_mesh_set_fast_prov_flags_iv_index(u8_t flags, u32_t iv_index);
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+#endif /* #ifndef _PROVISIONER_PROV_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_proxy.c b/components/bt/ble_mesh/mesh_core/provisioner_proxy.c
new file mode 100644
index 0000000000..d1b553cc87
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_proxy.c
@@ -0,0 +1,606 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "sdkconfig.h"
+
+#if CONFIG_BT_MESH
+#include "mesh_bearer_adapt.h"
+#include "mesh_trace.h"
+
+#include "net.h"
+#include "beacon.h"
+#include "foundation.h"
+#include "provisioner_prov.h"
+#include "provisioner_proxy.h"
+#include "provisioner_beacon.h"
+
+#if CONFIG_BT_MESH_PROVISIONER
+
+#define PDU_TYPE(data) (data[0] & BIT_MASK(6))
+#define PDU_SAR(data) (data[0] >> 6)
+
+#define SAR_COMPLETE 0x00
+#define SAR_FIRST 0x01
+#define SAR_CONT 0x02
+#define SAR_LAST 0x03
+
+#define CFG_FILTER_SET 0x00
+#define CFG_FILTER_ADD 0x01
+#define CFG_FILTER_REMOVE 0x02
+#define CFG_FILTER_STATUS 0x03
+
+#define PDU_HDR(sar, type) (sar << 6 | (type & BIT_MASK(6)))
+
+#define SERVER_BUF_SIZE 68
+
+#define ID_TYPE_NET 0x00
+#define ID_TYPE_NODE 0x01
+
+#define NODE_ID_LEN 19
+#define NET_ID_LEN 11
+
+#define CLOSE_REASON_PROXY 0xFF
+
+#define CONFIG_BT_MAX_CONN CONFIG_BT_ACL_CONNECTIONS
+
+static int conn_count;
+
+static struct bt_mesh_proxy_server {
+ struct bt_conn *conn;
+ /* Provisioner can use filter to double check the dst within mesh messages */
+ u16_t filter[CONFIG_BT_MESH_PROXY_FILTER_SIZE];
+ enum __packed {
+ NONE,
+ WHITELIST,
+ BLACKLIST,
+ PROV,
+ } filter_type;
+ u8_t msg_type;
+ struct net_buf_simple buf;
+ u8_t server_buf_data[SERVER_BUF_SIZE];
+} servers[CONFIG_BT_MAX_CONN];
+
+static struct bt_mesh_proxy_server *find_server(struct bt_conn *conn)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(servers); i++) {
+ if (servers[i].conn == conn) {
+ return &servers[i];
+ }
+ }
+
+ return NULL;
+}
+
+static int filter_status(struct bt_mesh_proxy_server *server,
+ struct net_buf_simple *buf)
+{
+ /* TODO: Deal with received proxy configuration status messages */
+ return 0;
+}
+
+#if 0
+static void send_filter_set(struct bt_mesh_proxy_server *server,
+ struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ /* TODO: Act as proxy client, send proxy configuration set messages */
+}
+
+static void send_filter_add(struct bt_mesh_proxy_server *server,
+ struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ /* TODO: Act as proxy client, send proxy configuration add messages */
+}
+
+static void send_filter_remove(struct bt_mesh_proxy_server *server,
+ struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ /* TODO: Act as proxy client, send proxy configuration remove messages */
+}
+#endif
+
+static void proxy_cfg(struct bt_mesh_proxy_server *server)
+{
+ struct net_buf_simple *buf = NET_BUF_SIMPLE(29);
+ struct bt_mesh_net_rx rx;
+ u8_t opcode;
+ int err;
+
+ /** In order to deal with proxy configuration messages, provisioner should
+ * do sth. like create mesh network after each device is provisioned.
+ */
+ err = bt_mesh_net_decode(&server->buf, BT_MESH_NET_IF_PROXY_CFG,
+ &rx, buf);
+ if (err) {
+ BT_ERR("Failed to decode Proxy Configuration (err %d)", err);
+ return;
+ }
+
+ /* Remove network headers */
+ net_buf_simple_pull(buf, BT_MESH_NET_HDR_LEN);
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (buf->len < 1) {
+ BT_WARN("Too short proxy configuration PDU");
+ return;
+ }
+
+ opcode = net_buf_simple_pull_u8(buf);
+ switch (opcode) {
+ case CFG_FILTER_STATUS:
+ filter_status(server, buf);
+ break;
+ default:
+ BT_WARN("Unhandled configuration OpCode 0x%02x", opcode);
+ break;
+ }
+}
+
+static void proxy_complete_pdu(struct bt_mesh_proxy_server *server)
+{
+ switch (server->msg_type) {
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ case BT_MESH_PROXY_NET_PDU:
+ BT_DBG("Mesh Network PDU");
+ bt_mesh_net_recv(&server->buf, 0, BT_MESH_NET_IF_PROXY);
+ break;
+ case BT_MESH_PROXY_BEACON:
+ BT_DBG("Mesh Beacon PDU");
+ provisioner_beacon_recv(&server->buf);
+ break;
+ case BT_MESH_PROXY_CONFIG:
+ BT_DBG("Mesh Configuration PDU");
+ proxy_cfg(server);
+ break;
+#endif
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ case BT_MESH_PROXY_PROV:
+ BT_DBG("Mesh Provisioning PDU");
+ provisioner_pb_gatt_recv(server->conn, &server->buf);
+ break;
+#endif
+ default:
+ BT_WARN("Unhandled Message Type 0x%02x", server->msg_type);
+ break;
+ }
+
+ net_buf_simple_init(&server->buf, 0);
+}
+
+#define ATTR_IS_PROV(uuid) (uuid == BT_UUID_MESH_PROV_VAL)
+
+static ssize_t proxy_recv(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr, const void *buf,
+ u16_t len, u16_t offset, u8_t flags)
+{
+ struct bt_mesh_proxy_server *server = find_server(conn);
+ const u8_t *data = buf;
+ u16_t srvc_uuid = 0;
+
+ if (!server) {
+ return -ENOTCONN;
+ }
+
+ if (len < 1) {
+ BT_WARN("Too small Proxy PDU");
+ return -EINVAL;
+ }
+
+ srvc_uuid = bt_mesh_gattc_get_service_uuid(conn);
+ if (!srvc_uuid) {
+ BT_ERR("No service uuid found");
+ return -ENOTCONN;
+ }
+
+ if (ATTR_IS_PROV(srvc_uuid) != (PDU_TYPE(data) == BT_MESH_PROXY_PROV)) {
+ BT_WARN("Proxy PDU type doesn't match GATT service uuid");
+ return -EINVAL;
+ }
+
+ if (len - 1 > net_buf_simple_tailroom(&server->buf)) {
+ BT_WARN("Too big proxy PDU");
+ return -EINVAL;
+ }
+
+ switch (PDU_SAR(data)) {
+ case SAR_COMPLETE:
+ if (server->buf.len) {
+ BT_WARN("Complete PDU while a pending incomplete one");
+ return -EINVAL;
+ }
+
+ server->msg_type = PDU_TYPE(data);
+ net_buf_simple_add_mem(&server->buf, data + 1, len - 1);
+ proxy_complete_pdu(server);
+ break;
+
+ case SAR_FIRST:
+ if (server->buf.len) {
+ BT_WARN("First PDU while a pending incomplete one");
+ return -EINVAL;
+ }
+
+ server->msg_type = PDU_TYPE(data);
+ net_buf_simple_add_mem(&server->buf, data + 1, len - 1);
+ break;
+
+ case SAR_CONT:
+ if (!server->buf.len) {
+ BT_WARN("Continuation with no prior data");
+ return -EINVAL;
+ }
+
+ if (server->msg_type != PDU_TYPE(data)) {
+ BT_WARN("Unexpected message type in continuation");
+ return -EINVAL;
+ }
+
+ net_buf_simple_add_mem(&server->buf, data + 1, len - 1);
+ break;
+
+ case SAR_LAST:
+ if (!server->buf.len) {
+ BT_WARN("Last SAR PDU with no prior data");
+ return -EINVAL;
+ }
+
+ if (server->msg_type != PDU_TYPE(data)) {
+ BT_WARN("Unexpected message type in last SAR PDU");
+ return -EINVAL;
+ }
+
+ net_buf_simple_add_mem(&server->buf, data + 1, len - 1);
+ proxy_complete_pdu(server);
+ break;
+ }
+
+ return len;
+}
+
+static void proxy_prov_connected(const u8_t addr[6], struct bt_conn *conn, int id)
+{
+ struct bt_mesh_proxy_server *server = NULL;
+
+ conn_count++;
+
+ if (!servers[id].conn) {
+ server = &servers[id];
+ }
+
+ if (!server) {
+ BT_ERR("No matching Proxy Client objects");
+ /** Disconnect current connection, clear part of prov_link
+ * information, like uuid, dev_addr, linking flag, etc.
+ */
+
+ return;
+ }
+
+ server->conn = bt_mesh_conn_ref(conn);
+ server->filter_type = NONE;
+ memset(server->filter, 0, sizeof(server->filter));
+ net_buf_simple_init(&server->buf, 0);
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (provisioner_set_prov_conn(addr, server->conn)) {
+ BT_ERR("%s: Fail to set prov_conn", __func__);
+ bt_mesh_gattc_disconnect(server->conn);
+ return;
+ }
+#endif
+
+ bt_mesh_gattc_exchange_mtu(id);
+}
+
+static void proxy_prov_disconnected(struct bt_conn *conn, u8_t reason)
+{
+ struct bt_mesh_proxy_server *server = NULL;
+ int i;
+
+ BT_DBG("conn %p, handle is %d, reason 0x%02x", conn, conn->handle, reason);
+
+ if (conn_count) {
+ conn_count--;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(servers); i++) {
+ server = &servers[i];
+ if (server->conn == conn) {
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ server->filter_type == PROV) {
+ provisioner_pb_gatt_close(conn, reason);
+ }
+ server->conn = NULL;
+ break;
+ }
+ }
+}
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+static ssize_t prov_write_ccc_descr(struct bt_conn *conn, u8_t *addr)
+{
+ struct bt_mesh_proxy_server *server;
+
+ server = find_server(conn);
+
+ if (!server) {
+ BT_ERR("No Proxy Server found");
+ return -ENOTCONN;
+ }
+
+ if (server->filter_type == NONE) {
+ server->filter_type = PROV;
+ return provisioner_pb_gatt_open(conn, addr);
+ }
+
+ return -EINVAL;
+}
+
+static ssize_t prov_notification(struct bt_conn *conn, u8_t *data, u16_t len)
+{
+ struct bt_mesh_proxy_server *server;
+
+ server = find_server(conn);
+
+ if (!server) {
+ BT_ERR("No Proxy Server found");
+ return -ENOTCONN;
+ }
+
+ if (server->filter_type == PROV) {
+ return proxy_recv(conn, NULL, data, len, 0, 0);
+ }
+
+ return -EINVAL;
+}
+
+int provisioner_pb_gatt_enable(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(servers); i++) {
+ if (servers[i].conn) {
+ servers[i].filter_type = PROV;
+ }
+ }
+
+ return 0;
+}
+
+int provisioner_pb_gatt_disable(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(servers); i++) {
+ struct bt_mesh_proxy_server *server = &servers[i];
+
+ if (server->conn && server->filter_type == PROV) {
+ bt_mesh_gattc_disconnect(server->conn);
+ server->filter_type = NONE;
+ }
+ }
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_PB_GATT */
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+static ssize_t proxy_write_ccc_descr(struct bt_conn *conn)
+{
+ struct bt_mesh_proxy_server *server;
+
+ server = find_server(conn);
+
+ if (!server) {
+ BT_ERR("No Proxy Server found");
+ return -ENOTCONN;
+ }
+
+ if (server->filter_type == NONE) {
+ server->filter_type = WHITELIST;
+ return 0;
+ }
+
+ return -EINVAL;
+}
+
+static ssize_t proxy_notification(struct bt_conn *conn, u8_t *data, u16_t len)
+{
+ return proxy_recv(conn, NULL, data, len, 0, 0);
+}
+
+/** Currently provisioner does't need bt_mesh_provisioner_proxy_enable()
+ * and bt_mesh_provisioner_proxy_disable() functions, and once they are
+ * used, provisioner can be enabled to parse node_id_adv and net_id_adv
+ * in order to support proxy client role.
+ * And if gatt_proxy is disabled, provisioner can stop dealing with
+ * these two kinds of connectable advertising packets.
+ */
+int bt_mesh_provisioner_proxy_enable(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(servers); i++) {
+ if (servers[i].conn) {
+ servers[i].filter_type = WHITELIST;
+ }
+ }
+
+ /** TODO: Once at leat one device has been provisioned, provisioner
+ * can be set to allow receiving and parsing node_id & net_id adv
+ * packets, and we may use a global flag to indicate this.
+ */
+
+ return 0;
+}
+
+static void bt_mesh_proxy_gatt_proxy_disconnect(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(servers); i++) {
+ struct bt_mesh_proxy_server *server = &servers[i];
+
+ if (server->conn && (server->filter_type == WHITELIST ||
+ server->filter_type == BLACKLIST)) {
+ server->filter_type = NONE;
+ bt_mesh_gattc_disconnect(server->conn);
+ }
+ }
+}
+
+int bt_mesh_provisioner_proxy_disable(void)
+{
+ BT_DBG("%s", __func__);
+
+ /** TODO: Once this function is invoked, provisioner shall stop
+ * receiving and parsing node_id & net_id adv packets, and if
+ * proxy connection exists, we should disconnect it.
+ */
+
+ bt_mesh_proxy_gatt_proxy_disconnect();
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_GATT_PROXY */
+
+static int proxy_send(struct bt_conn *conn, const void *data, u16_t len)
+{
+ BT_DBG("%u bytes: %s", len, bt_hex(data, len));
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY) || defined(CONFIG_BT_MESH_PB_GATT)
+ return bt_mesh_gattc_write_no_rsp(conn, NULL, data, len);
+#endif
+
+ return 0;
+}
+
+static int proxy_prov_segment_and_send(struct bt_conn *conn, u8_t type,
+ struct net_buf_simple *msg)
+{
+ u16_t mtu;
+
+ BT_DBG("conn %p type 0x%02x len %u: %s", conn, type, msg->len,
+ bt_hex(msg->data, msg->len));
+
+ mtu = bt_mesh_gattc_get_mtu_info(conn);
+ if (!mtu) {
+ BT_ERR("Conn used to get mtu does not exist");
+ return -ENOTCONN;
+ }
+
+ /* ATT_MTU - OpCode (1 byte) - Handle (2 bytes) */
+ mtu -= 3;
+ if (mtu > msg->len) {
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_COMPLETE, type));
+ return proxy_send(conn, msg->data, msg->len);
+ }
+
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_FIRST, type));
+ proxy_send(conn, msg->data, mtu);
+ net_buf_simple_pull(msg, mtu);
+
+ while (msg->len) {
+ if (msg->len + 1 < mtu) {
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_LAST, type));
+ proxy_send(conn, msg->data, msg->len);
+ break;
+ }
+
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_CONT, type));
+ proxy_send(conn, msg->data, mtu);
+ net_buf_simple_pull(msg, mtu);
+ }
+
+ return 0;
+}
+
+int provisioner_proxy_send(struct bt_conn *conn, u8_t type,
+ struct net_buf_simple *msg)
+{
+ struct bt_mesh_proxy_server *server = find_server(conn);
+
+ if (!server) {
+ BT_ERR("No Proxy Server found");
+ return -ENOTCONN;
+ }
+
+ if ((server->filter_type == PROV) != (type == BT_MESH_PROXY_PROV)) {
+ BT_ERR("Invalid PDU type for Proxy Client");
+ return -EINVAL;
+ }
+
+ return proxy_prov_segment_and_send(conn, type, msg);
+}
+
+static struct bt_prov_conn_cb conn_callbacks = {
+ .connected = proxy_prov_connected,
+ .disconnected = proxy_prov_disconnected,
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ .prov_write_descr = prov_write_ccc_descr,
+ .prov_notify = prov_notification,
+#endif /* CONFIG_BT_MESH_PB_GATT */
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ .proxy_write_descr = proxy_write_ccc_descr,
+ .proxy_notify = proxy_notification,
+#endif /* CONFIG_BT_MESH_GATT_PROXY */
+};
+
+void provisioner_proxy_srv_data_recv(struct net_buf_simple *buf)
+{
+ /** TODO: Parse node_id_adv or net_id_adv pkts. Currently we
+ * don't support this function, and if realized later, proxy
+ * client need to check if there is server structure left
+ * before create connection with a server.
+ * check conn_count & CONFIG_BT_MESH_PBG_SAME_TIME
+ */
+}
+
+int provisioner_proxy_init(void)
+{
+ int i;
+
+ /* Initialize the server receive buffers */
+ for (i = 0; i < ARRAY_SIZE(servers); i++) {
+ struct bt_mesh_proxy_server *server = &servers[i];
+ server->conn = NULL;
+ server->buf.size = SERVER_BUF_SIZE;
+ }
+
+ bt_mesh_gattc_conn_cb_register(&conn_callbacks);
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_PROVISIONER */
+
+#endif /* CONFIG_BT_MESH */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_core/provisioner_proxy.h b/components/bt/ble_mesh/mesh_core/provisioner_proxy.h
new file mode 100644
index 0000000000..474382918a
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/provisioner_proxy.h
@@ -0,0 +1,89 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _PROVISIONER_PROXY_H_
+#define _PROVISIONER_PROXY_H_
+
+#include "mesh_buf.h"
+
+#define BT_MESH_PROXY_NET_PDU 0x00
+#define BT_MESH_PROXY_BEACON 0x01
+#define BT_MESH_PROXY_CONFIG 0x02
+#define BT_MESH_PROXY_PROV 0x03
+
+/**
+ * @brief This function is called to send proxy protocol messages.
+ *
+ * @param[in] conn: Pointer to bt_conn structure
+ * @param[in] type: Proxy protocol message type
+ * @param[in] msg: Pointer to the buffer contains sending message.
+ *
+ * @return Zero-success, other-fail
+ */
+int provisioner_proxy_send(struct bt_conn *conn, u8_t type, struct net_buf_simple *msg);
+
+/**
+ * @brief This function is called to parse received node identity and net
+ * id adv pkts and create connection if deceided to.
+ *
+ * @param[in] buf: Pointer to the buffer contains received message.
+ *
+ * @return None
+ */
+void provisioner_proxy_srv_data_recv(struct net_buf_simple *buf);
+
+/**
+ * @brief This function is called to initialize proxy provisioner structure
+ * and register proxy connection related callbacks.
+ *
+ * @return Zero-success, other-fail
+ */
+int provisioner_proxy_init(void);
+
+/**
+ * @brief This function is called to enable dealing with proxy provisioning
+ * messages.
+ *
+ * @return Zero-success, other-fail
+ */
+int provisioner_pb_gatt_enable(void);
+
+/**
+ * @brief This function is called to disable dealing with proxy provisioning
+ * messages and if proxy provisioning connections exist, the connections
+ * will be disconnected.
+ *
+ * @return Zero-success, other-fail
+ */
+int provisioner_pb_gatt_disable(void);
+
+/* The following APIs are for application use */
+/**
+ * @brief This function is called to enable receiving node identity and net
+ * id adv pkts.
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_provisioner_proxy_enable(void);
+
+/**
+ * @brief This function is called to disable receiving node identity and net
+ * id adv pkts, and if proxy connections exist, these connections will
+ * be disconnected.
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_provisioner_proxy_disable(void);
+
+#endif /* _PROVISIONER_PROXY_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/proxy.c b/components/bt/ble_mesh/mesh_core/proxy.c
new file mode 100644
index 0000000000..ded6941f5d
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/proxy.c
@@ -0,0 +1,1313 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#include
+#include
+
+#include "mesh_buf.h"
+#include "mesh_util.h"
+#include "sdkconfig.h"
+#if CONFIG_BT_MESH
+
+#include "mesh_bearer_adapt.h"
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_PROXY)
+#include "mesh_trace.h"
+
+#include "mesh.h"
+#include "adv.h"
+#include "net.h"
+#include "prov.h"
+#include "beacon.h"
+#include "foundation.h"
+#include "access.h"
+#include "proxy.h"
+
+#if CONFIG_BT_MESH_NODE
+
+#define PDU_TYPE(data) (data[0] & BIT_MASK(6))
+#define PDU_SAR(data) (data[0] >> 6)
+
+#define SAR_COMPLETE 0x00
+#define SAR_FIRST 0x01
+#define SAR_CONT 0x02
+#define SAR_LAST 0x03
+
+#define CFG_FILTER_SET 0x00
+#define CFG_FILTER_ADD 0x01
+#define CFG_FILTER_REMOVE 0x02
+#define CFG_FILTER_STATUS 0x03
+
+#define CONFIG_BT_MAX_CONN CONFIG_BT_ACL_CONNECTIONS
+
+#define PDU_HDR(sar, type) (sar << 6 | (type & BIT_MASK(6)))
+
+#define CLIENT_BUF_SIZE 68
+
+static const struct bt_le_adv_param slow_adv_param = {
+ .options = (BT_LE_ADV_OPT_CONNECTABLE | BT_LE_ADV_OPT_ONE_TIME),
+ .interval_min = BT_GAP_ADV_SLOW_INT_MIN,
+ .interval_max = BT_GAP_ADV_SLOW_INT_MAX,
+};
+
+static const struct bt_le_adv_param fast_adv_param = {
+ .options = (BT_LE_ADV_OPT_CONNECTABLE | BT_LE_ADV_OPT_ONE_TIME),
+ .interval_min = BT_GAP_ADV_FAST_INT_MIN_0,
+ .interval_max = BT_GAP_ADV_FAST_INT_MAX_0,
+};
+
+static bool proxy_adv_enabled;
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+static void proxy_send_beacons(struct k_work *work);
+static u16_t proxy_ccc_val;
+#endif
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+static u16_t prov_ccc_val;
+static bool prov_fast_adv;
+#endif
+
+enum {
+ SAR_TIMER_START, /* Timer for SAR transfer has been started */
+ NUM_FLAGS,
+};
+
+#define PROXY_SAR_TRANS_TIMEOUT K_SECONDS(20)
+
+static struct bt_mesh_proxy_client {
+ /* Proxy Server: 20s timeout for each segmented proxy pdu */
+ ATOMIC_DEFINE(flags, NUM_FLAGS);
+ struct k_delayed_work sar_timer;
+ struct bt_conn *conn;
+ u16_t filter[CONFIG_BT_MESH_PROXY_FILTER_SIZE];
+ enum __packed {
+ NONE,
+ WHITELIST,
+ BLACKLIST,
+ PROV,
+ } filter_type;
+ u8_t msg_type;
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ struct k_work send_beacons;
+#endif
+ struct net_buf_simple buf;
+ u8_t buf_data[CLIENT_BUF_SIZE];
+} clients[CONFIG_BT_MAX_CONN] = {
+ [0 ... (CONFIG_BT_MAX_CONN - 1)] = {
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ .send_beacons = _K_WORK_INITIALIZER(proxy_send_beacons),
+#endif
+ .buf.size = CLIENT_BUF_SIZE,
+ },
+};
+
+/* Track which service is enabled */
+static enum {
+ MESH_GATT_NONE,
+ MESH_GATT_PROV,
+ MESH_GATT_PROXY,
+} gatt_svc = MESH_GATT_NONE;
+
+static char device_name[DEVICE_NAME_SIZE] = "ESP-BLE-MESH";
+
+int bt_mesh_set_device_name(const char *name)
+{
+ if (!name) {
+ BT_ERR("%s: name is NULL", __func__);
+ return -EINVAL;
+ }
+
+ if (strlen(name) > DEVICE_NAME_SIZE) {
+ BT_ERR("%s: too long name", __func__);
+ return -EINVAL;
+ }
+
+ memset(device_name, 0x0, sizeof(device_name));
+ memcpy(device_name, name, strlen(name));
+
+ return 0;
+}
+
+static struct bt_mesh_proxy_client *find_client(struct bt_conn *conn)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ if (clients[i].conn == conn) {
+ return &clients[i];
+ }
+ }
+
+ return NULL;
+}
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+/* Next subnet in queue to be advertised */
+static int next_idx;
+
+static int proxy_segment_and_send(struct bt_conn *conn, u8_t type,
+ struct net_buf_simple *msg);
+
+static int filter_set(struct bt_mesh_proxy_client *client,
+ struct net_buf_simple *buf)
+{
+ u8_t type;
+
+ if (buf->len < 1) {
+ BT_WARN("Too short Filter Set message");
+ return -EINVAL;
+ }
+
+ type = net_buf_simple_pull_u8(buf);
+ BT_DBG("type 0x%02x", type);
+
+ switch (type) {
+ case 0x00:
+ memset(client->filter, 0, sizeof(client->filter));
+ client->filter_type = WHITELIST;
+ break;
+ case 0x01:
+ memset(client->filter, 0, sizeof(client->filter));
+ client->filter_type = BLACKLIST;
+ break;
+ default:
+ BT_WARN("Prohibited Filter Type 0x%02x", type);
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+static void filter_add(struct bt_mesh_proxy_client *client, u16_t addr)
+{
+ int i;
+
+ BT_DBG("addr 0x%04x", addr);
+
+ if (addr == BT_MESH_ADDR_UNASSIGNED) {
+ return;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(client->filter); i++) {
+ if (client->filter[i] == addr) {
+ return;
+ }
+ }
+
+ for (i = 0; i < ARRAY_SIZE(client->filter); i++) {
+ if (client->filter[i] == BT_MESH_ADDR_UNASSIGNED) {
+ client->filter[i] = addr;
+ return;
+ }
+ }
+}
+
+static void filter_remove(struct bt_mesh_proxy_client *client, u16_t addr)
+{
+ int i;
+
+ BT_DBG("addr 0x%04x", addr);
+
+ if (addr == BT_MESH_ADDR_UNASSIGNED) {
+ return;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(client->filter); i++) {
+ if (client->filter[i] == addr) {
+ client->filter[i] = BT_MESH_ADDR_UNASSIGNED;
+ return;
+ }
+ }
+}
+
+static void send_filter_status(struct bt_mesh_proxy_client *client,
+ struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ struct bt_mesh_net_tx tx = {
+ .sub = rx->sub,
+ .ctx = &rx->ctx,
+ .src = bt_mesh_primary_addr(),
+ };
+ u16_t filter_size;
+ int i, err;
+
+ /* Configuration messages always have dst unassigned */
+ tx.ctx->addr = BT_MESH_ADDR_UNASSIGNED;
+
+ net_buf_simple_init(buf, 10);
+
+ net_buf_simple_add_u8(buf, CFG_FILTER_STATUS);
+
+ if (client->filter_type == WHITELIST) {
+ net_buf_simple_add_u8(buf, 0x00);
+ } else {
+ net_buf_simple_add_u8(buf, 0x01);
+ }
+
+ for (filter_size = 0, i = 0; i < ARRAY_SIZE(client->filter); i++) {
+ if (client->filter[i] != BT_MESH_ADDR_UNASSIGNED) {
+ filter_size++;
+ }
+ }
+
+ net_buf_simple_add_be16(buf, filter_size);
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ err = bt_mesh_net_encode(&tx, buf, true);
+ if (err) {
+ BT_ERR("Encoding Proxy cfg message failed (err %d)", err);
+ return;
+ }
+
+ err = proxy_segment_and_send(client->conn, BT_MESH_PROXY_CONFIG, buf);
+ if (err) {
+ BT_ERR("Failed to send proxy cfg message (err %d)", err);
+ }
+}
+
+static void proxy_cfg(struct bt_mesh_proxy_client *client)
+{
+ struct net_buf_simple *buf = NET_BUF_SIMPLE(29);
+ struct bt_mesh_net_rx rx;
+ u8_t opcode;
+ int err;
+
+ err = bt_mesh_net_decode(&client->buf, BT_MESH_NET_IF_PROXY_CFG,
+ &rx, buf);
+ if (err) {
+ BT_ERR("Failed to decode Proxy Configuration (err %d)", err);
+ return;
+ }
+
+ /* Remove network headers */
+ net_buf_simple_pull(buf, BT_MESH_NET_HDR_LEN);
+
+ BT_DBG("%u bytes: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (buf->len < 1) {
+ BT_WARN("Too short proxy configuration PDU");
+ return;
+ }
+
+ opcode = net_buf_simple_pull_u8(buf);
+ switch (opcode) {
+ case CFG_FILTER_SET:
+ filter_set(client, buf);
+ send_filter_status(client, &rx, buf);
+ break;
+ case CFG_FILTER_ADD:
+ while (buf->len >= 2) {
+ u16_t addr;
+
+ addr = net_buf_simple_pull_be16(buf);
+ filter_add(client, addr);
+ }
+ send_filter_status(client, &rx, buf);
+ break;
+ case CFG_FILTER_REMOVE:
+ while (buf->len >= 2) {
+ u16_t addr;
+
+ addr = net_buf_simple_pull_be16(buf);
+ filter_remove(client, addr);
+ }
+ send_filter_status(client, &rx, buf);
+ break;
+ default:
+ BT_WARN("Unhandled configuration OpCode 0x%02x", opcode);
+ break;
+ }
+}
+
+static int beacon_send(struct bt_conn *conn, struct bt_mesh_subnet *sub)
+{
+ struct net_buf_simple *buf = NET_BUF_SIMPLE(23);
+
+ net_buf_simple_init(buf, 1);
+ bt_mesh_beacon_create(sub, buf);
+
+ return proxy_segment_and_send(conn, BT_MESH_PROXY_BEACON, buf);
+}
+
+static void proxy_send_beacons(struct k_work *work)
+{
+ struct bt_mesh_proxy_client *client;
+ int i;
+
+ client = CONTAINER_OF(work, struct bt_mesh_proxy_client, send_beacons);
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx != BT_MESH_KEY_UNUSED) {
+ beacon_send(client->conn, sub);
+ }
+ }
+}
+
+void bt_mesh_proxy_beacon_send(struct bt_mesh_subnet *sub)
+{
+ int i;
+
+ if (!sub) {
+ /* NULL means we send on all subnets */
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ if (bt_mesh.sub[i].net_idx != BT_MESH_KEY_UNUSED) {
+ bt_mesh_proxy_beacon_send(&bt_mesh.sub[i]);
+ }
+ }
+
+ return;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ if (clients[i].conn) {
+ beacon_send(clients[i].conn, sub);
+ }
+ }
+}
+
+void bt_mesh_proxy_identity_start(struct bt_mesh_subnet *sub)
+{
+ sub->node_id = BT_MESH_NODE_IDENTITY_RUNNING;
+ sub->node_id_start = k_uptime_get_32();
+
+ /* Prioritize the recently enabled subnet */
+ next_idx = sub - bt_mesh.sub;
+}
+
+void bt_mesh_proxy_identity_stop(struct bt_mesh_subnet *sub)
+{
+ sub->node_id = BT_MESH_NODE_IDENTITY_STOPPED;
+ sub->node_id_start = 0;
+}
+
+int bt_mesh_proxy_identity_enable(void)
+{
+ int i, count = 0;
+
+ BT_DBG("%s", __func__);
+
+ if (!bt_mesh_is_provisioned()) {
+ return -EAGAIN;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ if (sub->node_id == BT_MESH_NODE_IDENTITY_NOT_SUPPORTED) {
+ continue;
+ }
+
+ bt_mesh_proxy_identity_start(sub);
+ count++;
+ }
+
+ if (count) {
+ bt_mesh_adv_update();
+ }
+
+ return 0;
+}
+
+#endif /* GATT_PROXY */
+
+static void proxy_recv_timeout(struct k_work *work)
+{
+ struct bt_mesh_proxy_client *client = NULL;
+
+ BT_DBG("%s", __func__);
+
+ client = CONTAINER_OF(work, struct bt_mesh_proxy_client, sar_timer.work);
+ if (!client || !client->conn) {
+ BT_ERR("%s, Client parameter is NULL", __func__);
+ return;
+ }
+
+ atomic_clear_bit(client->flags, SAR_TIMER_START);
+ net_buf_simple_init(&client->buf, 0);
+ bt_mesh_gatts_disconnect(client->conn, 0x13);
+}
+
+static void proxy_complete_pdu(struct bt_mesh_proxy_client *client)
+{
+ switch (client->msg_type) {
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ case BT_MESH_PROXY_NET_PDU:
+ BT_DBG("Mesh Network PDU");
+ bt_mesh_net_recv(&client->buf, 0, BT_MESH_NET_IF_PROXY);
+ break;
+ case BT_MESH_PROXY_BEACON:
+ BT_DBG("Mesh Beacon PDU");
+ bt_mesh_beacon_recv(&client->buf);
+ break;
+ case BT_MESH_PROXY_CONFIG:
+ BT_DBG("Mesh Configuration PDU");
+ proxy_cfg(client);
+ break;
+#endif
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ case BT_MESH_PROXY_PROV:
+ BT_DBG("Mesh Provisioning PDU");
+ bt_mesh_pb_gatt_recv(client->conn, &client->buf);
+ break;
+#endif
+ default:
+ BT_WARN("Unhandled Message Type 0x%02x", client->msg_type);
+ break;
+ }
+
+ net_buf_simple_init(&client->buf, 0);
+}
+
+#define ATTR_IS_PROV(attr) (attr->user_data != NULL)
+
+static ssize_t proxy_recv(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr, const void *buf,
+ u16_t len, u16_t offset, u8_t flags)
+{
+ struct bt_mesh_proxy_client *client = find_client(conn);
+ const u8_t *data = buf;
+
+ if (!client) {
+ return -ENOTCONN;
+ }
+
+ if (len < 1) {
+ BT_WARN("Too small Proxy PDU");
+ return -EINVAL;
+ }
+
+ if (ATTR_IS_PROV(attr) != (PDU_TYPE(data) == BT_MESH_PROXY_PROV)) {
+ BT_WARN("Proxy PDU type doesn't match GATT service");
+ return -EINVAL;
+ }
+
+ if (len - 1 > net_buf_simple_tailroom(&client->buf)) {
+ BT_WARN("Too big proxy PDU");
+ return -EINVAL;
+ }
+
+ switch (PDU_SAR(data)) {
+ case SAR_COMPLETE:
+ if (client->buf.len) {
+ BT_WARN("Complete PDU while a pending incomplete one");
+ return -EINVAL;
+ }
+
+ client->msg_type = PDU_TYPE(data);
+ net_buf_simple_add_mem(&client->buf, data + 1, len - 1);
+ proxy_complete_pdu(client);
+ break;
+
+ case SAR_FIRST:
+ if (client->buf.len) {
+ BT_WARN("First PDU while a pending incomplete one");
+ return -EINVAL;
+ }
+
+ if (!atomic_test_and_set_bit(client->flags, SAR_TIMER_START)) {
+ k_delayed_work_submit(&client->sar_timer, PROXY_SAR_TRANS_TIMEOUT);
+ }
+
+ client->msg_type = PDU_TYPE(data);
+ net_buf_simple_add_mem(&client->buf, data + 1, len - 1);
+ break;
+
+ case SAR_CONT:
+ if (!client->buf.len) {
+ BT_WARN("Continuation with no prior data");
+ return -EINVAL;
+ }
+
+ if (client->msg_type != PDU_TYPE(data)) {
+ BT_WARN("Unexpected message type in continuation");
+ return -EINVAL;
+ }
+
+ net_buf_simple_add_mem(&client->buf, data + 1, len - 1);
+ break;
+
+ case SAR_LAST:
+ if (!client->buf.len) {
+ BT_WARN("Last SAR PDU with no prior data");
+ return -EINVAL;
+ }
+
+ if (client->msg_type != PDU_TYPE(data)) {
+ BT_WARN("Unexpected message type in last SAR PDU");
+ return -EINVAL;
+ }
+
+ if (atomic_test_and_clear_bit(client->flags, SAR_TIMER_START)) {
+ k_delayed_work_cancel(&client->sar_timer);
+ }
+
+ net_buf_simple_add_mem(&client->buf, data + 1, len - 1);
+ proxy_complete_pdu(client);
+ break;
+ }
+
+ return len;
+}
+
+static int conn_count;
+
+static void proxy_connected(struct bt_conn *conn, u8_t err)
+{
+ struct bt_mesh_proxy_client *client;
+ int i;
+
+ BT_DBG("conn %p err 0x%02x", conn, err);
+
+ conn_count++;
+
+ /* Since we use ADV_OPT_ONE_TIME */
+ proxy_adv_enabled = false;
+
+ /* Try to re-enable advertising in case it's possible */
+ if (conn_count < CONFIG_BT_MAX_CONN) {
+ bt_mesh_adv_update();
+ }
+
+ for (client = NULL, i = 0; i < ARRAY_SIZE(clients); i++) {
+ if (!clients[i].conn) {
+ client = &clients[i];
+ break;
+ }
+ }
+
+ if (!client) {
+ BT_ERR("No free Proxy Client objects");
+ return;
+ }
+
+ client->conn = bt_mesh_conn_ref(conn);
+ client->filter_type = NONE;
+ memset(client->filter, 0, sizeof(client->filter));
+ net_buf_simple_init(&client->buf, 0);
+}
+
+static void proxy_disconnected(struct bt_conn *conn, u8_t reason)
+{
+ int i;
+
+ BT_DBG("conn %p reason 0x%02x", conn, reason);
+
+ conn_count--;
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ struct bt_mesh_proxy_client *client = &clients[i];
+
+ if (client->conn == conn) {
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT) &&
+ client->filter_type == PROV) {
+ bt_mesh_pb_gatt_close(conn);
+ }
+
+ if (atomic_test_and_clear_bit(client->flags, SAR_TIMER_START)) {
+ k_delayed_work_cancel(&client->sar_timer);
+ }
+ // Should reset the atomic flag after disconneted
+ bt_mesh_conn_unref(client->conn);
+ client->conn = NULL;
+ break;
+ }
+ }
+
+ bt_mesh_adv_update();
+}
+
+struct net_buf_simple *bt_mesh_proxy_get_buf(void)
+{
+ struct net_buf_simple *buf = &clients[0].buf;
+
+ net_buf_simple_init(buf, 0);
+
+ return buf;
+}
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+static ssize_t prov_ccc_write(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ const void *buf, u16_t len,
+ u16_t offset, u8_t flags)
+{
+ struct bt_mesh_proxy_client *client;
+ u16_t *value = attr->user_data;
+
+ BT_DBG("len %u: %s", len, bt_hex(buf, len));
+
+ if (len != sizeof(*value)) {
+ return BT_GATT_ERR(BT_ATT_ERR_INVALID_ATTRIBUTE_LEN);
+ }
+
+ *value = sys_get_le16(buf);
+ if (*value != BT_GATT_CCC_NOTIFY) {
+ BT_WARN("Client wrote 0x%04x instead enabling notify", *value);
+ return len;
+ }
+
+ /* If a connection exists there must be a client */
+ client = find_client(conn);
+ __ASSERT(client, "No client for connection");
+
+ if (client->filter_type == NONE) {
+ client->filter_type = PROV;
+ bt_mesh_pb_gatt_open(conn);
+ }
+
+ return len;
+}
+
+static ssize_t prov_ccc_read(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ void *buf, u16_t len, u16_t offset)
+{
+ u16_t *value = attr->user_data;
+
+ return bt_mesh_gatts_attr_read(conn, attr, buf, len, offset, value,
+ sizeof(*value));
+}
+
+/* Mesh Provisioning Service Declaration */
+static struct bt_gatt_attr prov_attrs[] = {
+ BT_GATT_PRIMARY_SERVICE(BT_UUID_MESH_PROV),
+
+ BT_GATT_CHARACTERISTIC(BT_UUID_MESH_PROV_DATA_IN,
+ BT_GATT_CHRC_WRITE_WITHOUT_RESP),
+ BT_GATT_DESCRIPTOR(BT_UUID_MESH_PROV_DATA_IN, BT_GATT_PERM_WRITE,
+ NULL, proxy_recv, (void *)1),
+
+ BT_GATT_CHARACTERISTIC(BT_UUID_MESH_PROV_DATA_OUT,
+ BT_GATT_CHRC_NOTIFY),
+ BT_GATT_DESCRIPTOR(BT_UUID_MESH_PROV_DATA_OUT, BT_GATT_PERM_NONE,
+ NULL, NULL, NULL),
+ /* Add custom CCC as clients need to be tracked individually */
+ BT_GATT_DESCRIPTOR(BT_UUID_GATT_CCC,
+ BT_GATT_PERM_WRITE | BT_GATT_PERM_READ,
+ prov_ccc_read, prov_ccc_write, &prov_ccc_val),
+};
+
+struct bt_gatt_service prov_svc = BT_GATT_SERVICE(prov_attrs);
+
+int bt_mesh_proxy_prov_enable(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ /**TODO: Change by Espressif. We should not register service here, otherwise will
+ * block btu task, so currently just start it. Need to further check*/
+ bt_mesh_gatts_service_start(&prov_svc);
+ //bt_mesh_gatts_service_register(&prov_svc);
+ gatt_svc = MESH_GATT_PROV;
+ prov_fast_adv = true;
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ if (clients[i].conn) {
+ clients[i].filter_type = PROV;
+ }
+ }
+
+
+ return 0;
+}
+
+int bt_mesh_proxy_prov_disable(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ /**TODO: Change by Espressif. Please refer to bt_mesh_proxy_prov_enable for
+ * more info */
+ bt_mesh_gatts_service_stop(&prov_svc);
+ //bt_mesh_gatts_service_unregister(&prov_svc);
+ gatt_svc = MESH_GATT_NONE;
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ struct bt_mesh_proxy_client *client = &clients[i];
+
+ if (client->conn && client->filter_type == PROV) {
+ bt_mesh_pb_gatt_close(client->conn);
+ client->filter_type = NONE;
+ }
+ }
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_PB_GATT */
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+static ssize_t proxy_ccc_write(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ const void *buf, u16_t len,
+ u16_t offset, u8_t flags)
+{
+ struct bt_mesh_proxy_client *client;
+ u16_t value;
+
+ BT_DBG("len %u: %s", len, bt_hex(buf, len));
+
+ if (len != sizeof(value)) {
+ return BT_GATT_ERR(BT_ATT_ERR_INVALID_ATTRIBUTE_LEN);
+ }
+
+ value = sys_get_le16(buf);
+ if (value != BT_GATT_CCC_NOTIFY) {
+ BT_WARN("Client wrote 0x%04x instead enabling notify", value);
+ return len;
+ }
+
+ /* If a connection exists there must be a client */
+ client = find_client(conn);
+ __ASSERT(client, "No client for connection");
+
+ if (client->filter_type == NONE) {
+ client->filter_type = WHITELIST;
+ k_work_submit(&client->send_beacons);
+ }
+
+ return len;
+}
+
+static ssize_t proxy_ccc_read(struct bt_conn *conn,
+ const struct bt_gatt_attr *attr,
+ void *buf, u16_t len, u16_t offset)
+{
+ u16_t *value = attr->user_data;
+
+ return bt_mesh_gatts_attr_read(conn, attr, buf, len, offset, value,
+ sizeof(*value));
+}
+
+/* Mesh Proxy Service Declaration */
+static struct bt_gatt_attr proxy_attrs[] = {
+ BT_GATT_PRIMARY_SERVICE(BT_UUID_MESH_PROXY),
+
+ BT_GATT_CHARACTERISTIC(BT_UUID_MESH_PROXY_DATA_IN,
+ BT_GATT_CHRC_WRITE_WITHOUT_RESP),
+ BT_GATT_DESCRIPTOR(BT_UUID_MESH_PROXY_DATA_IN, BT_GATT_PERM_WRITE,
+ NULL, proxy_recv, NULL),
+
+ BT_GATT_CHARACTERISTIC(BT_UUID_MESH_PROXY_DATA_OUT,
+ BT_GATT_CHRC_NOTIFY),
+ BT_GATT_DESCRIPTOR(BT_UUID_MESH_PROXY_DATA_OUT, BT_GATT_PERM_NONE,
+ NULL, NULL, NULL),
+ /* Add custom CCC as clients need to be tracked individually */
+ BT_GATT_DESCRIPTOR(BT_UUID_GATT_CCC,
+ BT_GATT_PERM_READ | BT_GATT_PERM_WRITE,
+ proxy_ccc_read, proxy_ccc_write, &proxy_ccc_val),
+};
+
+struct bt_gatt_service proxy_svc = BT_GATT_SERVICE(proxy_attrs);
+
+int bt_mesh_proxy_gatt_enable(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ /**TODO: We should not register service here, otherwise will
+ * block btu task, so currently just start it. Need to further check*/
+ bt_mesh_gatts_service_start(&proxy_svc);
+ //bt_mesh_gatts_service_register(&proxy_svc);
+ gatt_svc = MESH_GATT_PROXY;
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ if (clients[i].conn) {
+ clients[i].filter_type = WHITELIST;
+ }
+ }
+
+ return 0;
+}
+
+void bt_mesh_proxy_gatt_disconnect(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ struct bt_mesh_proxy_client *client = &clients[i];
+
+ if (client->conn && (client->filter_type == WHITELIST ||
+ client->filter_type == BLACKLIST)) {
+ client->filter_type = NONE;
+ bt_mesh_gatts_disconnect(client->conn, 0x13);
+ //BT_HCI_ERR_REMOTE_USER_TERM_CONN);
+ }
+ }
+}
+
+int bt_mesh_proxy_gatt_disable(void)
+{
+ BT_DBG("%s", __func__);
+
+ bt_mesh_proxy_gatt_disconnect();
+
+ /**TODO: please refer to comment of bt_mesh_proxy_gatt_enable,
+ * We not de-register, but only stop service*/
+ bt_mesh_gatts_service_stop(&proxy_svc);
+ //bt_mesh_gatts_service_unregister(&proxy_svc);
+ gatt_svc = MESH_GATT_NONE;
+
+ return 0;
+}
+
+void bt_mesh_proxy_addr_add(struct net_buf_simple *buf, u16_t addr)
+{
+ struct bt_mesh_proxy_client *client =
+ CONTAINER_OF(buf, struct bt_mesh_proxy_client, buf);
+
+ BT_DBG("filter_type %u addr 0x%04x", client->filter_type, addr);
+
+ if (client->filter_type == WHITELIST) {
+ filter_add(client, addr);
+ } else if (client->filter_type == BLACKLIST) {
+ filter_remove(client, addr);
+ }
+}
+
+static bool client_filter_match(struct bt_mesh_proxy_client *client,
+ u16_t addr)
+{
+ int i;
+
+ BT_DBG("filter_type %u addr 0x%04x", client->filter_type, addr);
+
+ if (client->filter_type == WHITELIST) {
+ for (i = 0; i < ARRAY_SIZE(client->filter); i++) {
+ if (client->filter[i] == addr) {
+ return true;
+ }
+ }
+
+ return false;
+ }
+
+ if (client->filter_type == BLACKLIST) {
+ for (i = 0; i < ARRAY_SIZE(client->filter); i++) {
+ if (client->filter[i] == addr) {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ return false;
+}
+
+bool bt_mesh_proxy_relay(struct net_buf_simple *buf, u16_t dst)
+{
+ bool relayed = false;
+ int i;
+
+ BT_DBG("%u bytes to dst 0x%04x", buf->len, dst);
+
+ for (i = 0; i < ARRAY_SIZE(clients); i++) {
+ struct bt_mesh_proxy_client *client = &clients[i];
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(32);
+
+ if (!client->conn) {
+ continue;
+ }
+
+ if (!client_filter_match(client, dst)) {
+ continue;
+ }
+
+ /* Proxy PDU sending modifies the original buffer,
+ * so we need to make a copy.
+ */
+ net_buf_simple_init(msg, 1);
+ net_buf_simple_add_mem(msg, buf->data, buf->len);
+
+ bt_mesh_proxy_send(client->conn, BT_MESH_PROXY_NET_PDU, msg);
+ relayed = true;
+ }
+
+ return relayed;
+}
+
+#endif /* CONFIG_BT_MESH_GATT_PROXY */
+
+static int proxy_send(struct bt_conn *conn, const void *data, u16_t len)
+{
+ BT_DBG("%u bytes: %s", len, bt_hex(data, len));
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ if (gatt_svc == MESH_GATT_PROXY) {
+ return bt_mesh_gatts_notify(conn, &proxy_attrs[4], data, len);
+ }
+#endif
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (gatt_svc == MESH_GATT_PROV) {
+ return bt_mesh_gatts_notify(conn, &prov_attrs[4], data, len);
+ }
+#endif
+
+ return 0;
+}
+
+static int proxy_segment_and_send(struct bt_conn *conn, u8_t type,
+ struct net_buf_simple *msg)
+{
+ u16_t mtu;
+
+ BT_DBG("conn %p type 0x%02x len %u: %s", conn, type, msg->len,
+ bt_hex(msg->data, msg->len));
+
+ /* ATT_MTU - OpCode (1 byte) - Handle (2 bytes) */
+ mtu = bt_mesh_gatt_get_mtu(conn) - 3;
+ if (mtu > msg->len) {
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_COMPLETE, type));
+ return proxy_send(conn, msg->data, msg->len);
+ }
+
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_FIRST, type));
+ proxy_send(conn, msg->data, mtu);
+ net_buf_simple_pull(msg, mtu);
+
+ while (msg->len) {
+ if (msg->len + 1 < mtu) {
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_LAST, type));
+ proxy_send(conn, msg->data, msg->len);
+ break;
+ }
+
+ net_buf_simple_push_u8(msg, PDU_HDR(SAR_CONT, type));
+ proxy_send(conn, msg->data, mtu);
+ net_buf_simple_pull(msg, mtu);
+ }
+
+ return 0;
+}
+
+int bt_mesh_proxy_send(struct bt_conn *conn, u8_t type,
+ struct net_buf_simple *msg)
+{
+ struct bt_mesh_proxy_client *client = find_client(conn);
+
+ if (!client) {
+ BT_ERR("No Proxy Client found");
+ return -ENOTCONN;
+ }
+
+ if ((client->filter_type == PROV) != (type == BT_MESH_PROXY_PROV)) {
+ BT_ERR("Invalid PDU type for Proxy Client");
+ return -EINVAL;
+ }
+
+ return proxy_segment_and_send(conn, type, msg);
+}
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+static u8_t prov_svc_data[20] = { 0x27, 0x18, };
+
+static const struct bt_data prov_ad[] = {
+ BT_DATA_BYTES(BT_DATA_FLAGS, (BT_LE_AD_GENERAL | BT_LE_AD_NO_BREDR)),
+ BT_DATA_BYTES(BT_DATA_UUID16_ALL, 0x27, 0x18),
+ BT_DATA(BT_DATA_SVC_DATA16, prov_svc_data, sizeof(prov_svc_data)),
+};
+
+static const struct bt_data prov_sd[] = {
+ BT_DATA(BT_DATA_NAME_COMPLETE, device_name,
+ (sizeof(device_name) - 1)),
+};
+#endif /* PB_GATT */
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+
+#define ID_TYPE_NET 0x00
+#define ID_TYPE_NODE 0x01
+
+#define NODE_ID_LEN 19
+#define NET_ID_LEN 11
+
+#define NODE_ID_TIMEOUT K_SECONDS(CONFIG_BT_MESH_NODE_ID_TIMEOUT)
+
+static u8_t proxy_svc_data[NODE_ID_LEN] = { 0x28, 0x18, };
+
+static const struct bt_data node_id_ad[] = {
+ BT_DATA_BYTES(BT_DATA_FLAGS, (BT_LE_AD_GENERAL | BT_LE_AD_NO_BREDR)),
+ BT_DATA_BYTES(BT_DATA_UUID16_ALL, 0x28, 0x18),
+ BT_DATA(BT_DATA_SVC_DATA16, proxy_svc_data, NODE_ID_LEN),
+};
+
+static const struct bt_data net_id_ad[] = {
+ BT_DATA_BYTES(BT_DATA_FLAGS, (BT_LE_AD_GENERAL | BT_LE_AD_NO_BREDR)),
+ BT_DATA_BYTES(BT_DATA_UUID16_ALL, 0x28, 0x18),
+ BT_DATA(BT_DATA_SVC_DATA16, proxy_svc_data, NET_ID_LEN),
+};
+
+static int node_id_adv(struct bt_mesh_subnet *sub)
+{
+ u8_t tmp[16];
+ int err;
+
+ BT_DBG("%s", __func__);
+
+ proxy_svc_data[2] = ID_TYPE_NODE;
+
+ err = bt_mesh_rand(proxy_svc_data + 11, 8);
+ if (err) {
+ return err;
+ }
+
+ memset(tmp, 0, 6);
+ memcpy(tmp + 6, proxy_svc_data + 11, 8);
+ sys_put_be16(bt_mesh_primary_addr(), tmp + 14);
+
+ err = bt_encrypt_be(sub->keys[sub->kr_flag].identity, tmp, tmp);
+ if (err) {
+ return err;
+ }
+
+ memcpy(proxy_svc_data + 3, tmp + 8, 8);
+
+ err = bt_le_adv_start(&fast_adv_param, node_id_ad,
+ ARRAY_SIZE(node_id_ad), NULL, 0);
+ if (err) {
+ BT_WARN("Failed to advertise using Node ID (err %d)", err);
+ return err;
+ }
+
+ proxy_adv_enabled = true;
+
+ return 0;
+}
+
+static int net_id_adv(struct bt_mesh_subnet *sub)
+{
+ int err;
+
+ BT_DBG("%s", __func__);
+
+ proxy_svc_data[2] = ID_TYPE_NET;
+
+ BT_DBG("Advertising with NetId %s",
+ bt_hex(sub->keys[sub->kr_flag].net_id, 8));
+
+ memcpy(proxy_svc_data + 3, sub->keys[sub->kr_flag].net_id, 8);
+
+ err = bt_le_adv_start(&slow_adv_param, net_id_ad,
+ ARRAY_SIZE(net_id_ad), NULL, 0);
+ if (err) {
+ BT_WARN("Failed to advertise using Network ID (err %d)", err);
+ return err;
+ }
+
+ proxy_adv_enabled = true;
+
+ return 0;
+}
+
+static bool advertise_subnet(struct bt_mesh_subnet *sub)
+{
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ return false;
+ }
+
+ return (sub->node_id == BT_MESH_NODE_IDENTITY_RUNNING ||
+ bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_ENABLED);
+}
+
+static struct bt_mesh_subnet *next_sub(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub;
+
+ sub = &bt_mesh.sub[(i + next_idx) % ARRAY_SIZE(bt_mesh.sub)];
+ if (advertise_subnet(sub)) {
+ next_idx = (next_idx + 1) % ARRAY_SIZE(bt_mesh.sub);
+ return sub;
+ }
+ }
+
+ return NULL;
+}
+
+static int sub_count(void)
+{
+ int i, count = 0;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+
+ if (advertise_subnet(sub)) {
+ count++;
+ }
+ }
+
+ return count;
+}
+
+static s32_t gatt_proxy_advertise(struct bt_mesh_subnet *sub)
+{
+ s32_t remaining = K_FOREVER;
+ int subnet_count;
+
+ BT_DBG("%s", __func__);
+
+ if (conn_count == CONFIG_BT_MAX_CONN) {
+ BT_DBG("Connectable advertising deferred (max connections)");
+ return remaining;
+ }
+
+ if (!sub) {
+ BT_WARN("No subnets to advertise on");
+ return remaining;
+ }
+
+ if (sub->node_id == BT_MESH_NODE_IDENTITY_RUNNING) {
+ u32_t active = k_uptime_get_32() - sub->node_id_start;
+
+ if (active < NODE_ID_TIMEOUT) {
+ remaining = NODE_ID_TIMEOUT - active;
+ BT_DBG("Node ID active for %u ms, %d ms remaining",
+ active, remaining);
+ node_id_adv(sub);
+ } else {
+ bt_mesh_proxy_identity_stop(sub);
+ BT_DBG("Node ID stopped");
+ }
+ }
+
+ if (sub->node_id == BT_MESH_NODE_IDENTITY_STOPPED) {
+ if (bt_mesh_gatt_proxy_get() == BT_MESH_GATT_PROXY_ENABLED) {
+ net_id_adv(sub);
+ } else {
+ return gatt_proxy_advertise(next_sub());
+ }
+ }
+
+ subnet_count = sub_count();
+ BT_DBG("sub_count %u", subnet_count);
+ if (subnet_count > 1) {
+ s32_t max_timeout;
+
+ /* We use NODE_ID_TIMEOUT as a starting point since it may
+ * be less than 60 seconds. Divide this period into at least
+ * 6 slices, but make sure that a slice is at least one
+ * second long (to avoid excessive rotation).
+ */
+ max_timeout = NODE_ID_TIMEOUT / max(subnet_count, 6);
+ max_timeout = max(max_timeout, K_SECONDS(1));
+
+ if (remaining > max_timeout || remaining < 0) {
+ remaining = max_timeout;
+ }
+ }
+
+ BT_DBG("Advertising %d ms for net_idx 0x%04x", remaining, sub->net_idx);
+
+ return remaining;
+}
+#endif /* GATT_PROXY */
+
+s32_t bt_mesh_proxy_adv_start(void)
+{
+ BT_DBG("%s", __func__);
+
+ if (gatt_svc == MESH_GATT_NONE) {
+ BT_DBG("%s, MESH_GATT_NONE", __func__);
+ return K_FOREVER;
+ }
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (!bt_mesh_is_provisioned()) {
+ const struct bt_le_adv_param *param;
+
+ if (prov_fast_adv) {
+ param = &fast_adv_param;
+ } else {
+ param = &slow_adv_param;
+ }
+
+ if (bt_le_adv_start(param, prov_ad, ARRAY_SIZE(prov_ad),
+ prov_sd, ARRAY_SIZE(prov_sd)) == 0) {
+ proxy_adv_enabled = true;
+
+ /* Advertise 60 seconds using fast interval */
+ if (prov_fast_adv) {
+ prov_fast_adv = false;
+ return K_SECONDS(60);
+ }
+ }
+ }
+#endif /* PB_GATT */
+
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ if (bt_mesh_is_provisioned()) {
+ return gatt_proxy_advertise(next_sub());
+ }
+#endif /* GATT_PROXY */
+
+ return K_FOREVER;
+}
+
+void bt_mesh_proxy_adv_stop(void)
+{
+ int err;
+
+ BT_DBG("adv_enabled %u", proxy_adv_enabled);
+
+ if (!proxy_adv_enabled) {
+ return;
+ }
+
+ err = bt_le_adv_stop();
+ if (err) {
+ BT_ERR("Failed to stop advertising (err %d)", err);
+ } else {
+ proxy_adv_enabled = false;
+ }
+}
+
+static struct bt_conn_cb conn_callbacks = {
+ .connected = proxy_connected,
+ .disconnected = proxy_disconnected,
+};
+
+int bt_mesh_proxy_init(void)
+{
+ for (int i = 0; i < ARRAY_SIZE(clients); i++) {
+ struct bt_mesh_proxy_client *client = &clients[i];
+ k_delayed_work_init(&client->sar_timer, proxy_recv_timeout);
+ }
+ bt_mesh_gatts_conn_cb_register(&conn_callbacks);
+
+#if defined(CONFIG_BT_MESH_PB_GATT)
+ if (!bt_mesh_prov_get_uuid()) {
+ BT_ERR("%s: prov->uuid not initialized", __func__);
+ return -EINVAL;
+ }
+ memcpy(prov_svc_data + 2, bt_mesh_prov_get_uuid(), 16);
+#endif
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_NODE */
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/proxy.h b/components/bt/ble_mesh/mesh_core/proxy.h
new file mode 100644
index 0000000000..fcf496d1f1
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/proxy.h
@@ -0,0 +1,51 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __PROXY_H__
+#define __PROXY_H__
+
+#include "net.h"
+#include "mesh_buf.h"
+#include "mesh_bearer_adapt.h"
+
+#define BT_MESH_PROXY_NET_PDU 0x00
+#define BT_MESH_PROXY_BEACON 0x01
+#define BT_MESH_PROXY_CONFIG 0x02
+#define BT_MESH_PROXY_PROV 0x03
+
+#define DEVICE_NAME_SIZE 29
+
+int bt_mesh_set_device_name(const char *name);
+
+int bt_mesh_proxy_send(struct bt_conn *conn, u8_t type,
+ struct net_buf_simple *msg);
+
+int bt_mesh_proxy_prov_enable(void);
+int bt_mesh_proxy_prov_disable(void);
+
+int bt_mesh_proxy_gatt_enable(void);
+int bt_mesh_proxy_gatt_disable(void);
+void bt_mesh_proxy_gatt_disconnect(void);
+
+void bt_mesh_proxy_beacon_send(struct bt_mesh_subnet *sub);
+
+struct net_buf_simple *bt_mesh_proxy_get_buf(void);
+
+s32_t bt_mesh_proxy_adv_start(void);
+void bt_mesh_proxy_adv_stop(void);
+
+void bt_mesh_proxy_identity_start(struct bt_mesh_subnet *sub);
+void bt_mesh_proxy_identity_stop(struct bt_mesh_subnet *sub);
+
+bool bt_mesh_proxy_relay(struct net_buf_simple *buf, u16_t dst);
+void bt_mesh_proxy_addr_add(struct net_buf_simple *buf, u16_t addr);
+
+int bt_mesh_proxy_init(void);
+
+#endif /* __PROXY_H__ */
diff --git a/components/bt/ble_mesh/mesh_core/settings.c b/components/bt/ble_mesh/mesh_core/settings.c
new file mode 100644
index 0000000000..3114915f01
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings.c
@@ -0,0 +1,1616 @@
+/*
+ * Copyright (c) 2018 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+
+#include "sdkconfig.h"
+
+#include "base64.h"
+#include "mesh_types.h"
+#include "mesh_util.h"
+#include "mesh_access.h"
+#include "mesh_main.h"
+
+#include "mesh_buf.h"
+#include "mesh_kernel.h"
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_SETTINGS)
+#include "mesh_trace.h"
+
+#include "mesh.h"
+#include "net.h"
+#include "crypto.h"
+#include "transport.h"
+#include "access.h"
+#include "foundation.h"
+#include "proxy.h"
+#include "cfg_srv.h"
+#include "settings.h"
+#include "settings/include/settings.h"
+
+#if CONFIG_BT_MESH_SETTINGS
+/* Tracking of what storage changes are pending for App and Net Keys. We
+ * track this in a separate array here instead of within the respective
+ * bt_mesh_app_key and bt_mesh_subnet structs themselves, since once a key
+ * gets deleted its struct becomes invalid and may be reused for other keys.
+ */
+static struct key_update {
+ u16_t key_idx: 12, /* AppKey or NetKey Index */
+ valid: 1, /* 1 if this entry is valid, 0 if not */
+ app_key: 1, /* 1 if this is an AppKey, 0 if a NetKey */
+ clear: 1; /* 1 if key needs clearing, 0 if storing */
+} key_updates[CONFIG_BT_MESH_APP_KEY_COUNT + CONFIG_BT_MESH_SUBNET_COUNT];
+
+static struct k_delayed_work pending_store;
+
+/* Mesh network storage information */
+struct net_val {
+ u16_t primary_addr;
+ u8_t dev_key[16];
+} __packed;
+
+/* Sequence number storage */
+struct seq_val {
+ u8_t val[3];
+} __packed;
+
+/* Heartbeat Publication storage */
+struct hb_pub_val {
+ u16_t dst;
+ u8_t period;
+ u8_t ttl;
+ u16_t feat;
+ u16_t net_idx: 12,
+ indefinite: 1;
+};
+
+/* Miscelaneous configuration server model states */
+struct cfg_val {
+ u8_t net_transmit;
+ u8_t relay;
+ u8_t relay_retransmit;
+ u8_t beacon;
+ u8_t gatt_proxy;
+ u8_t frnd;
+ u8_t default_ttl;
+};
+
+/* IV Index & IV Update storage */
+struct iv_val {
+ u32_t iv_index;
+ bool iv_update;
+} __packed;
+
+/* Replay Protection List storage */
+struct rpl_val {
+ u32_t seq: 24,
+ old_iv: 1;
+};
+
+/* NetKey storage information */
+struct net_key_val {
+ u8_t kr_flag: 1,
+ kr_phase: 7;
+ u8_t val[2][16];
+} __packed;
+
+/* AppKey storage information */
+struct app_key_val {
+ u16_t net_idx;
+ bool updated;
+ u8_t val[2][16];
+} __packed;
+
+struct mod_pub_val {
+ u16_t addr;
+ u16_t key;
+ u8_t ttl;
+ u8_t retransmit;
+ u8_t period;
+ u8_t period_div: 4,
+ cred: 1;
+};
+
+/* We need this so we don't overwrite app-hardcoded values in case FCB
+ * contains a history of changes but then has a NULL at the end.
+ */
+static struct {
+ bool valid;
+ struct cfg_val cfg;
+} stored_cfg;
+
+static int net_set(int argc, char **argv, char *val)
+{
+ struct net_val net;
+ int len, err;
+
+ BT_DBG("val %s", val ? val : "(null)");
+
+ if (!val) {
+ bt_mesh_comp_unprovision();
+ memset(bt_mesh.dev_key, 0, sizeof(bt_mesh.dev_key));
+ return 0;
+ }
+
+ len = sizeof(net);
+ err = settings_bytes_from_str(val, &net, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(net)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len, sizeof(net));
+ return -EINVAL;
+ }
+
+ memcpy(bt_mesh.dev_key, net.dev_key, sizeof(bt_mesh.dev_key));
+ bt_mesh_comp_provision(net.primary_addr);
+
+ BT_DBG("Provisioned with primary address 0x%04x", net.primary_addr);
+ BT_DBG("Recovered DevKey %s", bt_hex(bt_mesh.dev_key, 16));
+
+ return 0;
+}
+
+static int iv_set(int argc, char **argv, char *val)
+{
+ struct iv_val iv;
+ int len, err;
+
+ BT_DBG("val %s", val ? val : "(null)");
+
+ if (!val) {
+ bt_mesh.iv_index = 0;
+ bt_mesh.iv_update = 0;
+ return 0;
+ }
+
+ len = sizeof(iv);
+ err = settings_bytes_from_str(val, &iv, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(iv)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len, sizeof(iv));
+ return -EINVAL;
+ }
+
+ bt_mesh.iv_index = iv.iv_index;
+ bt_mesh.iv_update = iv.iv_update;
+
+ BT_DBG("IV Index 0x%04x (IV Update Flag %u)", bt_mesh.iv_index,
+ bt_mesh.iv_update);
+
+ return 0;
+}
+
+static int seq_set(int argc, char **argv, char *val)
+{
+ struct seq_val seq;
+ int len, err;
+
+ BT_DBG("val %s", val ? val : "(null)");
+
+ if (!val) {
+ bt_mesh.seq = 0;
+ return 0;
+ }
+
+ len = sizeof(seq);
+ err = settings_bytes_from_str(val, &seq, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(seq)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len, sizeof(seq));
+ return -EINVAL;
+ }
+
+ bt_mesh.seq = ((u32_t)seq.val[0] | ((u32_t)seq.val[1] << 8) |
+ ((u32_t)seq.val[2] << 16));
+
+ if (CONFIG_BT_MESH_SEQ_STORE_RATE > 0) {
+ /* Make sure we have a large enough sequence number. We
+ * subtract 1 so that the first transmission causes a write
+ * to the settings storage.
+ */
+ bt_mesh.seq += (CONFIG_BT_MESH_SEQ_STORE_RATE -
+ (bt_mesh.seq % CONFIG_BT_MESH_SEQ_STORE_RATE));
+ bt_mesh.seq--;
+ }
+
+ BT_DBG("Sequence Number 0x%06x", bt_mesh.seq);
+
+ return 0;
+}
+
+static struct bt_mesh_rpl *rpl_find(u16_t src)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.rpl); i++) {
+ if (bt_mesh.rpl[i].src == src) {
+ return &bt_mesh.rpl[i];
+ }
+ }
+
+ return NULL;
+}
+
+static struct bt_mesh_rpl *rpl_alloc(u16_t src)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.rpl); i++) {
+ if (!bt_mesh.rpl[i].src) {
+ bt_mesh.rpl[i].src = src;
+ return &bt_mesh.rpl[i];
+ }
+ }
+
+ return NULL;
+}
+
+static int rpl_set(int argc, char **argv, char *val)
+{
+ struct bt_mesh_rpl *entry;
+ struct rpl_val rpl;
+ int len, err;
+ u16_t src;
+
+ if (argc < 1) {
+ BT_ERR("Invalid argc (%d)", argc);
+ return -ENOENT;
+ }
+
+ BT_DBG("argv[0] %s val %s", argv[0], val ? val : "(null)");
+
+ src = strtol(argv[0], NULL, 16);
+ entry = rpl_find(src);
+
+ if (!val) {
+ if (entry) {
+ memset(entry, 0, sizeof(*entry));
+ } else {
+ BT_WARN("Unable to find RPL entry for 0x%04x", src);
+ }
+
+ return 0;
+ }
+
+ if (!entry) {
+ entry = rpl_alloc(src);
+ if (!entry) {
+ BT_ERR("Unable to allocate RPL entry for 0x%04x", src);
+ return -ENOMEM;
+ }
+ }
+
+ len = sizeof(rpl);
+ err = settings_bytes_from_str(val, &rpl, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(rpl)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len, sizeof(rpl));
+ return -EINVAL;
+ }
+
+ entry->seq = rpl.seq;
+ entry->old_iv = rpl.old_iv;
+
+ BT_DBG("RPL entry for 0x%04x: Seq 0x%06x old_iv %u", entry->src,
+ entry->seq, entry->old_iv);
+
+ return 0;
+}
+
+static int net_key_set(int argc, char **argv, char *val)
+{
+ struct bt_mesh_subnet *sub;
+ struct net_key_val key;
+ int len, i, err;
+ u16_t net_idx;
+
+ BT_DBG("argv[0] %s val %s", argv[0], val ? val : "(null)");
+
+ net_idx = strtol(argv[0], NULL, 16);
+ sub = bt_mesh_subnet_get(net_idx);
+
+ if (!val) {
+ if (!sub) {
+ BT_ERR("No subnet with NetKeyIndex 0x%03x", net_idx);
+ return -ENOENT;
+ }
+
+ BT_DBG("Deleting NetKeyIndex 0x%03x", net_idx);
+ //TODO: Should update the zephyr code v1.12.0 of the cfg_srv.c implementation.
+ //bt_mesh_subnet_del(sub);
+ return 0;
+ }
+
+ len = sizeof(key);
+ err = settings_bytes_from_str(val, &key, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(key)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len, sizeof(key));
+ return -EINVAL;
+ }
+
+ if (sub) {
+ BT_DBG("Updating existing NetKeyIndex 0x%03x", net_idx);
+
+ sub->kr_flag = key.kr_flag;
+ sub->kr_phase = key.kr_phase;
+ memcpy(sub->keys[0].net, &key.val[0], 16);
+ memcpy(sub->keys[1].net, &key.val[1], 16);
+
+ return 0;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ if (bt_mesh.sub[i].net_idx == BT_MESH_KEY_UNUSED) {
+ sub = &bt_mesh.sub[i];
+ break;
+ }
+ }
+
+ if (!sub) {
+ BT_ERR("No space to allocate a new subnet");
+ return -ENOMEM;
+ }
+
+ sub->net_idx = net_idx;
+ sub->kr_flag = key.kr_flag;
+ sub->kr_phase = key.kr_phase;
+ memcpy(sub->keys[0].net, &key.val[0], 16);
+ memcpy(sub->keys[1].net, &key.val[1], 16);
+
+ BT_DBG("NetKeyIndex 0x%03x recovered from storage", net_idx);
+
+ return 0;
+}
+
+static int app_key_set(int argc, char **argv, char *val)
+{
+ struct bt_mesh_app_key *app;
+ struct bt_mesh_subnet *sub;
+ struct app_key_val key;
+ u16_t app_idx;
+ int len, err;
+
+ BT_DBG("argv[0] %s val %s", argv[0], val ? val : "(null)");
+
+ app_idx = strtol(argv[0], NULL, 16);
+
+ if (!val) {
+ BT_DBG("Deleting AppKeyIndex 0x%03x", app_idx);
+
+ app = bt_mesh_app_key_find(app_idx);
+ if (app) {
+ //TODO: Should update the zephyr code v1.12.0 of the cfg_srv.c implementation.
+ //bt_mesh_app_key_del(app);
+ }
+
+ return 0;
+ }
+
+ len = sizeof(key);
+ err = settings_bytes_from_str(val, &key, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(key)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len, sizeof(key));
+ return -EINVAL;
+ }
+
+ sub = bt_mesh_subnet_get(key.net_idx);
+ if (!sub) {
+ BT_ERR("Failed to find subnet 0x%03x", key.net_idx);
+ return -ENOENT;
+ }
+
+ app = bt_mesh_app_key_find(app_idx);
+ if (!app) {
+ app = bt_mesh_app_key_alloc(app_idx);
+ }
+
+ if (!app) {
+ BT_ERR("No space for a new app key");
+ return -ENOMEM;
+ }
+
+ app->net_idx = key.net_idx;
+ app->app_idx = app_idx;
+ app->updated = key.updated;
+ memcpy(app->keys[0].val, key.val[0], 16);
+ memcpy(app->keys[1].val, key.val[1], 16);
+
+ bt_mesh_app_id(app->keys[0].val, &app->keys[0].id);
+ bt_mesh_app_id(app->keys[1].val, &app->keys[1].id);
+
+ BT_DBG("AppKeyIndex 0x%03x recovered from storage", app_idx);
+
+ return 0;
+}
+
+static int hb_pub_set(int argc, char **argv, char *val)
+{
+ struct bt_mesh_hb_pub *pub = bt_mesh_hb_pub_get();
+ struct hb_pub_val hb_val;
+ int len, err;
+
+ BT_DBG("val %s", val ? val : "(null)");
+
+ if (!pub) {
+ return -ENOENT;
+ }
+
+ if (!val) {
+ pub->dst = BT_MESH_ADDR_UNASSIGNED;
+ pub->count = 0;
+ pub->ttl = 0;
+ pub->period = 0;
+ pub->feat = 0;
+
+ BT_DBG("Cleared heartbeat publication");
+ return 0;
+ }
+
+ len = sizeof(hb_val);
+ err = settings_bytes_from_str(val, &hb_val, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(hb_val)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len,
+ sizeof(hb_val));
+ return -EINVAL;
+ }
+
+ pub->dst = hb_val.dst;
+ pub->period = hb_val.period;
+ pub->ttl = hb_val.ttl;
+ pub->feat = hb_val.feat;
+ pub->net_idx = hb_val.net_idx;
+
+ if (hb_val.indefinite) {
+ pub->count = 0xffff;
+ } else {
+ pub->count = 0;
+ }
+
+ BT_DBG("Restored heartbeat publication");
+
+ return 0;
+}
+
+static int cfg_set(int argc, char **argv, char *val)
+{
+ struct bt_mesh_cfg_srv *cfg = bt_mesh_cfg_get();
+ int len, err;
+
+ BT_DBG("val %s", val ? val : "(null)");
+
+ if (!cfg) {
+ return -ENOENT;
+ }
+
+ if (!val) {
+ stored_cfg.valid = false;
+ BT_DBG("Cleared configuration state");
+ return 0;
+ }
+
+ len = sizeof(stored_cfg.cfg);
+ err = settings_bytes_from_str(val, &stored_cfg.cfg, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return err;
+ }
+
+ if (len != sizeof(stored_cfg.cfg)) {
+ BT_ERR("Unexpected value length (%d != %zu)", len,
+ sizeof(stored_cfg.cfg));
+ return -EINVAL;
+ }
+
+ stored_cfg.valid = true;
+ BT_DBG("Restored configuration state");
+
+ return 0;
+}
+
+static int mod_set_bind(struct bt_mesh_model *mod, char *val)
+{
+ int len, err, i;
+
+ /* Start with empty array regardless of cleared or set value */
+ for (i = 0; i < ARRAY_SIZE(mod->keys); i++) {
+ mod->keys[i] = BT_MESH_KEY_UNUSED;
+ }
+
+ if (!val) {
+ BT_DBG("Cleared bindings for model");
+ return 0;
+ }
+
+ len = sizeof(mod->keys);
+ err = settings_bytes_from_str(val, mod->keys, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return -EINVAL;
+ }
+
+ BT_DBG("Decoded %u bound keys for model", len / sizeof(mod->keys[0]));
+ return 0;
+}
+
+static int mod_set_sub(struct bt_mesh_model *mod, char *val)
+{
+ int len, err;
+
+ /* Start with empty array regardless of cleared or set value */
+ memset(mod->groups, 0, sizeof(mod->groups));
+
+ if (!val) {
+ BT_DBG("Cleared subscriptions for model");
+ return 0;
+ }
+
+ len = sizeof(mod->groups);
+ err = settings_bytes_from_str(val, mod->groups, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return -EINVAL;
+ }
+
+ BT_DBG("Decoded %u subscribed group addresses for model",
+ len / sizeof(mod->groups[0]));
+ return 0;
+}
+
+static int mod_set_pub(struct bt_mesh_model *mod, char *val)
+{
+ struct mod_pub_val pub;
+ int len, err;
+
+ if (!mod->pub) {
+ BT_WARN("Model has no publication context!");
+ return -EINVAL;
+ }
+
+ if (!val) {
+ mod->pub->addr = BT_MESH_ADDR_UNASSIGNED;
+ mod->pub->key = 0;
+ mod->pub->cred = 0;
+ mod->pub->ttl = 0;
+ mod->pub->period = 0;
+ mod->pub->retransmit = 0;
+ mod->pub->count = 0;
+
+ BT_DBG("Cleared publication for model");
+ return 0;
+ }
+
+ len = sizeof(pub);
+ err = settings_bytes_from_str(val, &pub, &len);
+ if (err) {
+ BT_ERR("Failed to decode value %s (err %d)", val, err);
+ return -EINVAL;
+ }
+
+ if (len != sizeof(pub)) {
+ BT_ERR("Invalid length for model publication");
+ return -EINVAL;
+ }
+
+ mod->pub->addr = pub.addr;
+ mod->pub->key = pub.key;
+ mod->pub->cred = pub.cred;
+ mod->pub->ttl = pub.ttl;
+ mod->pub->period = pub.period;
+ mod->pub->retransmit = pub.retransmit;
+ mod->pub->count = 0;
+
+ BT_DBG("Restored model publication, dst 0x%04x app_idx 0x%03x",
+ pub.addr, pub.key);
+
+ return 0;
+}
+
+static int mod_set(bool vnd, int argc, char **argv, char *val)
+{
+ struct bt_mesh_model *mod;
+ u8_t elem_idx, mod_idx;
+ u16_t mod_key;
+
+ if (argc < 2) {
+ BT_ERR("Too small argc (%d)", argc);
+ return -ENOENT;
+ }
+
+ mod_key = strtol(argv[0], NULL, 16);
+ elem_idx = mod_key >> 8;
+ mod_idx = mod_key;
+
+ BT_DBG("Decoded mod_key 0x%04x as elem_idx %u mod_idx %u",
+ mod_key, elem_idx, mod_idx);
+
+ mod = bt_mesh_model_get(vnd, elem_idx, mod_idx);
+ if (!mod) {
+ BT_ERR("Failed to get model for elem_idx %u mod_idx %u",
+ elem_idx, mod_idx);
+ return -ENOENT;
+ }
+
+ if (!strcmp(argv[1], "bind")) {
+ return mod_set_bind(mod, val);
+ }
+
+ if (!strcmp(argv[1], "sub")) {
+ return mod_set_sub(mod, val);
+ }
+
+ if (!strcmp(argv[1], "pub")) {
+ return mod_set_pub(mod, val);
+ }
+
+ BT_WARN("Unknown module key %s", argv[1]);
+ return -ENOENT;
+}
+
+static int sig_mod_set(int argc, char **argv, char *val)
+{
+ return mod_set(false, argc, argv, val);
+}
+
+static int vnd_mod_set(int argc, char **argv, char *val)
+{
+ return mod_set(true, argc, argv, val);
+}
+
+const struct mesh_setting {
+ const char *name;
+ int (*func)(int argc, char **argv, char *val);
+} settings[] = {
+ { "Net", net_set },
+ { "IV", iv_set },
+ { "Seq", seq_set },
+ { "RPL", rpl_set },
+ { "NetKey", net_key_set },
+ { "AppKey", app_key_set },
+ { "HBPub", hb_pub_set },
+ { "Cfg", cfg_set },
+ { "s", sig_mod_set },
+ { "v", vnd_mod_set },
+};
+
+static int mesh_set(int argc, char **argv, char *val)
+{
+ int i;
+
+ if (argc < 1) {
+ BT_ERR("Insufficient number of arguments");
+ return -EINVAL;
+ }
+
+ BT_DBG("argv[0] %s val %s", argv[0], val ? val : "(null)");
+
+ for (i = 0; i < ARRAY_SIZE(settings); i++) {
+ if (!strcmp(settings[i].name, argv[0])) {
+ argc--;
+ argv++;
+
+ return settings[i].func(argc, argv, val);
+ }
+ }
+
+ BT_WARN("No matching handler for key %s", argv[0]);
+
+ return -ENOENT;
+}
+
+static int subnet_init(struct bt_mesh_subnet *sub)
+{
+ int err;
+
+ err = bt_mesh_net_keys_create(&sub->keys[0], sub->keys[0].net);
+ if (err) {
+ BT_ERR("Unable to generate keys for subnet");
+ return -EIO;
+ }
+
+ if (sub->kr_phase != BT_MESH_KR_NORMAL) {
+ err = bt_mesh_net_keys_create(&sub->keys[1], sub->keys[1].net);
+ if (err) {
+ BT_ERR("Unable to generate keys for subnet");
+ memset(&sub->keys[0], 0, sizeof(sub->keys[0]));
+ return -EIO;
+ }
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_GATT_PROXY)) {
+ sub->node_id = BT_MESH_NODE_IDENTITY_STOPPED;
+ } else {
+ sub->node_id = BT_MESH_NODE_IDENTITY_NOT_SUPPORTED;
+ }
+
+ /* Make sure we have valid beacon data to be sent */
+ bt_mesh_net_beacon_update(sub);
+
+ return 0;
+}
+
+static void commit_mod(struct bt_mesh_model *mod, struct bt_mesh_elem *elem,
+ bool vnd, bool primary, void *user_data)
+{
+ if (mod->pub && mod->pub->update &&
+ mod->pub->addr != BT_MESH_ADDR_UNASSIGNED) {
+ s32_t ms = bt_mesh_model_pub_period_get(mod);
+ if (ms) {
+ BT_DBG("Starting publish timer (period %u ms)", ms);
+ k_delayed_work_submit(&mod->pub->timer, ms);
+ }
+ }
+}
+
+static int mesh_commit(void)
+{
+ struct bt_mesh_hb_pub *hb_pub;
+ struct bt_mesh_cfg_srv *cfg;
+ int i;
+
+ BT_DBG("sub[0].net_idx 0x%03x", bt_mesh.sub[0].net_idx);
+
+ if (bt_mesh.sub[0].net_idx == BT_MESH_KEY_UNUSED) {
+ /* Nothing to do since we're not yet provisioned */
+ return 0;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_PB_GATT)) {
+#if defined(CONFIG_BT_MESH_NODE)
+ bt_mesh_proxy_prov_disable();
+#endif
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.sub); i++) {
+ struct bt_mesh_subnet *sub = &bt_mesh.sub[i];
+ int err;
+
+ if (sub->net_idx == BT_MESH_KEY_UNUSED) {
+ continue;
+ }
+
+ err = subnet_init(sub);
+ if (err) {
+ BT_ERR("Failed to init subnet 0x%03x", sub->net_idx);
+ }
+ }
+
+ /* Set initial IV Update procedure state time-stamp */
+ bt_mesh.last_update = BT_MESH_NET_IVU_UNKNOWN;
+
+ /* Set a timer to transition back to normal mode */
+ if (bt_mesh.iv_update) {
+ k_delayed_work_submit(&bt_mesh.ivu_complete,
+ BT_MESH_NET_IVU_TIMEOUT);
+ }
+
+ bt_mesh_model_foreach(commit_mod, NULL);
+
+ hb_pub = bt_mesh_hb_pub_get();
+ if (hb_pub && hb_pub->dst != BT_MESH_ADDR_UNASSIGNED &&
+ hb_pub->count && hb_pub->period) {
+ BT_DBG("Starting heartbeat publication");
+ k_work_submit(&hb_pub->timer.work);
+ }
+
+ cfg = bt_mesh_cfg_get();
+ if (cfg && stored_cfg.valid) {
+ cfg->net_transmit = stored_cfg.cfg.net_transmit;
+ cfg->relay = stored_cfg.cfg.relay;
+ cfg->relay_retransmit = stored_cfg.cfg.relay_retransmit;
+ cfg->beacon = stored_cfg.cfg.beacon;
+ cfg->gatt_proxy = stored_cfg.cfg.gatt_proxy;
+ cfg->frnd = stored_cfg.cfg.frnd;
+ cfg->default_ttl = stored_cfg.cfg.default_ttl;
+ }
+
+ bt_mesh.valid = 1;
+
+#if defined(CONFIG_BT_MESH_NODE)
+ bt_mesh_net_start();
+#endif
+
+ return 0;
+}
+
+//BT_SETTINGS_DEFINE(mesh, mesh_set, mesh_commit, NULL);
+static struct settings_handler mesh = {
+ .name = "bt_mesh",
+ .h_set = mesh_set,
+ .h_commit = mesh_commit,
+ .h_export = NULL,
+};
+
+static void schedule_store(int flag)
+{
+ s32_t timeout;
+
+ atomic_set_bit(bt_mesh.flags, flag);
+
+ if (atomic_test_bit(bt_mesh.flags, BT_MESH_NET_PENDING) ||
+ atomic_test_bit(bt_mesh.flags, BT_MESH_IV_PENDING) ||
+ atomic_test_bit(bt_mesh.flags, BT_MESH_SEQ_PENDING)) {
+ timeout = K_NO_WAIT;
+ } else if (atomic_test_bit(bt_mesh.flags, BT_MESH_RPL_PENDING) &&
+ (CONFIG_BT_MESH_RPL_STORE_TIMEOUT <
+ CONFIG_BT_MESH_STORE_TIMEOUT)) {
+ timeout = K_SECONDS(CONFIG_BT_MESH_RPL_STORE_TIMEOUT);
+ } else {
+ timeout = K_SECONDS(CONFIG_BT_MESH_STORE_TIMEOUT);
+ }
+
+ BT_DBG("Waiting %d seconds", timeout / MSEC_PER_SEC);
+
+ k_delayed_work_submit(&pending_store, timeout);
+}
+
+static void clear_iv(void)
+{
+ BT_DBG("Clearing IV");
+ settings_save_one("bt_mesh/IV", NULL);
+}
+
+static void clear_net(void)
+{
+ BT_DBG("Clearing Network");
+ settings_save_one("bt_mesh/Net", NULL);
+}
+
+static void store_pending_net(void)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct net_val))];
+ struct net_val net;
+ char *str;
+
+ BT_DBG("addr 0x%04x DevKey %s", bt_mesh_primary_addr(),
+ bt_hex(bt_mesh.dev_key, 16));
+
+ net.primary_addr = bt_mesh_primary_addr();
+ memcpy(net.dev_key, bt_mesh.dev_key, 16);
+
+ str = settings_str_from_bytes(&net, sizeof(net), buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode Network as value");
+ return;
+ }
+
+ BT_DBG("Saving Network as value %s", str);
+ settings_save_one("bt_mesh/Net", str);
+}
+
+void bt_mesh_store_net(void)
+{
+ schedule_store(BT_MESH_NET_PENDING);
+}
+
+static void store_pending_iv(void)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct iv_val))];
+ struct iv_val iv;
+ char *str;
+
+ iv.iv_index = bt_mesh.iv_index;
+ iv.iv_update = bt_mesh.iv_update;
+
+ str = settings_str_from_bytes(&iv, sizeof(iv), buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode IV as value");
+ return;
+ }
+
+ BT_DBG("Saving IV as value %s", str);
+ settings_save_one("bt_mesh/IV", str);
+}
+
+void bt_mesh_store_iv(void)
+{
+ schedule_store(BT_MESH_IV_PENDING);
+ /* Always update Seq whenever IV changes */
+ schedule_store(BT_MESH_SEQ_PENDING);
+}
+
+static void store_pending_seq(void)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct seq_val))];
+ struct seq_val seq;
+ char *str;
+
+ seq.val[0] = bt_mesh.seq;
+ seq.val[1] = bt_mesh.seq >> 8;
+ seq.val[2] = bt_mesh.seq >> 16;
+
+ str = settings_str_from_bytes(&seq, sizeof(seq), buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode Seq as value");
+ return;
+ }
+
+ BT_DBG("Saving Seq as value %s", str);
+ settings_save_one("bt_mesh/Seq", str);
+}
+
+void bt_mesh_store_seq(void)
+{
+ if (CONFIG_BT_MESH_SEQ_STORE_RATE &&
+ (bt_mesh.seq % CONFIG_BT_MESH_SEQ_STORE_RATE)) {
+ return;
+ }
+
+ schedule_store(BT_MESH_SEQ_PENDING);
+}
+
+static void store_rpl(struct bt_mesh_rpl *entry)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct rpl_val))];
+ struct rpl_val rpl;
+ char path[18];
+ char *str;
+
+ BT_DBG("src 0x%04x seq 0x%06x old_iv %u", entry->src, entry->seq,
+ entry->old_iv);
+
+ rpl.seq = entry->seq;
+ rpl.old_iv = entry->old_iv;
+
+ str = settings_str_from_bytes(&rpl, sizeof(rpl), buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode RPL as value");
+ return;
+ }
+
+ snprintf(path, sizeof(path), "bt_mesh/RPL/%x", entry->src);
+
+ BT_DBG("Saving RPL %s as value %s", path, str);
+ settings_save_one(path, str);
+}
+
+static void clear_rpl(void)
+{
+ int i;
+
+ BT_DBG("");
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.rpl); i++) {
+ struct bt_mesh_rpl *rpl = &bt_mesh.rpl[i];
+ char path[18];
+
+ if (!rpl->src) {
+ continue;
+ }
+
+ snprintf(path, sizeof(path), "bt_mesh/RPL/%x", rpl->src);
+ settings_save_one(path, NULL);
+
+ memset(rpl, 0, sizeof(*rpl));
+ }
+}
+
+static void store_pending_rpl(void)
+{
+ int i;
+
+ BT_DBG("");
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.rpl); i++) {
+ struct bt_mesh_rpl *rpl = &bt_mesh.rpl[i];
+
+ if (rpl->store) {
+ rpl->store = false;
+ store_rpl(rpl);
+ }
+ }
+}
+
+static void store_pending_hb_pub(void)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct hb_pub_val))];
+ struct bt_mesh_hb_pub *pub = bt_mesh_hb_pub_get();
+ struct hb_pub_val val;
+ char *str;
+
+ if (!pub) {
+ return;
+ }
+
+ if (pub->dst == BT_MESH_ADDR_UNASSIGNED) {
+ str = NULL;
+ } else {
+ val.indefinite = (pub->count = 0xffff);
+ val.dst = pub->dst;
+ val.period = pub->period;
+ val.ttl = pub->ttl;
+ val.feat = pub->feat;
+ val.net_idx = pub->net_idx;
+
+ str = settings_str_from_bytes(&val, sizeof(val),
+ buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode hb pub as value");
+ return;
+ }
+ }
+
+ BT_DBG("Saving Heartbeat Publication as value %s",
+ str ? str : "(null)");
+ settings_save_one("bt_mesh/HBPub", str);
+}
+
+static void store_pending_cfg(void)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct cfg_val))];
+ struct bt_mesh_cfg_srv *cfg = bt_mesh_cfg_get();
+ struct cfg_val val;
+ char *str;
+
+ if (!cfg) {
+ return;
+ }
+
+ val.net_transmit = cfg->net_transmit;
+ val.relay = cfg->relay;
+ val.relay_retransmit = cfg->relay_retransmit;
+ val.beacon = cfg->beacon;
+ val.gatt_proxy = cfg->gatt_proxy;
+ val.frnd = cfg->frnd;
+ val.default_ttl = cfg->default_ttl;
+
+ str = settings_str_from_bytes(&val, sizeof(val), buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode configuration as value");
+ return;
+ }
+
+ BT_DBG("Saving configuration as value %s", str);
+ settings_save_one("bt_mesh/Cfg", str);
+}
+
+static void clear_cfg(void)
+{
+ BT_DBG("Clearing configuration");
+ settings_save_one("bt_mesh/Cfg", NULL);
+}
+
+static void clear_app_key(u16_t app_idx)
+{
+ char path[20];
+
+ BT_DBG("AppKeyIndex 0x%03x", app_idx);
+
+ snprintf(path, sizeof(path), "bt_mesh/AppKey/%x", app_idx);
+ settings_save_one(path, NULL);
+}
+
+static void clear_net_key(u16_t net_idx)
+{
+ char path[20];
+
+ BT_DBG("NetKeyIndex 0x%03x", net_idx);
+
+ snprintf(path, sizeof(path), "bt_mesh/NetKey/%x", net_idx);
+ settings_save_one(path, NULL);
+}
+
+static void store_net_key(struct bt_mesh_subnet *sub)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct net_key_val))];
+ struct net_key_val key;
+ char path[20];
+ char *str;
+
+ BT_DBG("NetKeyIndex 0x%03x NetKey %s", sub->net_idx,
+ bt_hex(sub->keys[0].net, 16));
+
+ memcpy(&key.val[0], sub->keys[0].net, 16);
+ memcpy(&key.val[1], sub->keys[1].net, 16);
+ key.kr_flag = sub->kr_flag;
+ key.kr_phase = sub->kr_phase;
+
+ str = settings_str_from_bytes(&key, sizeof(key), buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode NetKey as value");
+ return;
+ }
+
+ snprintf(path, sizeof(path), "bt_mesh/NetKey/%x", sub->net_idx);
+
+ BT_DBG("Saving NetKey %s as value %s", path, str);
+ settings_save_one(path, str);
+}
+
+static void store_app_key(struct bt_mesh_app_key *app)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct app_key_val))];
+ struct app_key_val key;
+ char path[20];
+ char *str;
+
+ key.net_idx = app->net_idx;
+ key.updated = app->updated;
+ memcpy(key.val[0], app->keys[0].val, 16);
+ memcpy(key.val[1], app->keys[1].val, 16);
+
+ str = settings_str_from_bytes(&key, sizeof(key), buf, sizeof(buf));
+ if (!str) {
+ BT_ERR("Unable to encode AppKey as value");
+ return;
+ }
+
+ snprintf(path, sizeof(path), "bt_mesh/AppKey/%x", app->app_idx);
+
+ BT_DBG("Saving AppKey %s as value %s", path, str);
+ settings_save_one(path, str);
+}
+
+static void store_pending_keys(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(key_updates); i++) {
+ struct key_update *update = &key_updates[i];
+
+ if (!update->valid) {
+ continue;
+ }
+
+ if (update->clear) {
+ if (update->app_key) {
+ clear_app_key(update->key_idx);
+ } else {
+ clear_net_key(update->key_idx);
+ }
+ } else {
+ if (update->app_key) {
+ struct bt_mesh_app_key *key;
+
+ key = bt_mesh_app_key_find(update->key_idx);
+ if (key) {
+ store_app_key(key);
+ } else {
+ BT_WARN("AppKeyIndex 0x%03x not found",
+ update->key_idx);
+ }
+
+ } else {
+ struct bt_mesh_subnet *sub;
+
+ sub = bt_mesh_subnet_get(update->key_idx);
+ if (sub) {
+ store_net_key(sub);
+ } else {
+ BT_WARN("NetKeyIndex 0x%03x not found",
+ update->key_idx);
+ }
+ }
+ }
+
+ update->valid = 0;
+ }
+}
+
+static void encode_mod_path(struct bt_mesh_model *mod, bool vnd,
+ const char *key, char *path, size_t path_len)
+{
+ // TODO:
+ u16_t mod_key = (((u16_t)mod->element_idx << 8) | mod->model_idx);
+
+ if (vnd) {
+ snprintf(path, path_len, "bt_mesh/v/%x/%s", mod_key, key);
+ } else {
+ snprintf(path, path_len, "bt_mesh/s/%x/%s", mod_key, key);
+ }
+}
+
+static void store_pending_mod_bind(struct bt_mesh_model *mod, bool vnd)
+{
+ u16_t keys[CONFIG_BT_MESH_MODEL_KEY_COUNT];
+ char buf[BT_SETTINGS_SIZE(sizeof(keys))];
+ char path[20];
+ int i, count;
+ char *val;
+
+ for (i = 0, count = 0; i < ARRAY_SIZE(mod->keys); i++) {
+ if (mod->keys[i] != BT_MESH_KEY_UNUSED) {
+ keys[count++] = mod->keys[i];
+ }
+ }
+
+ if (count) {
+ val = settings_str_from_bytes(keys, count * sizeof(keys[0]),
+ buf, sizeof(buf));
+ if (!val) {
+ BT_ERR("Unable to encode model bindings as value");
+ return;
+ }
+ } else {
+ val = NULL;
+ }
+
+ encode_mod_path(mod, vnd, "bind", path, sizeof(path));
+
+ BT_DBG("Saving %s as %s", path, val ? val : "(null)");
+ settings_save_one(path, val);
+}
+
+static void store_pending_mod_sub(struct bt_mesh_model *mod, bool vnd)
+{
+ u16_t groups[CONFIG_BT_MESH_MODEL_GROUP_COUNT];
+ char buf[BT_SETTINGS_SIZE(sizeof(groups))];
+ char path[20];
+ int i, count;
+ char *val;
+
+ for (i = 0, count = 0; i < ARRAY_SIZE(mod->groups); i++) {
+ if (mod->groups[i] != BT_MESH_ADDR_UNASSIGNED) {
+ groups[count++] = mod->groups[i];
+ }
+ }
+
+ if (count) {
+ val = settings_str_from_bytes(groups, count * sizeof(groups[0]),
+ buf, sizeof(buf));
+ if (!val) {
+ BT_ERR("Unable to encode model subscription as value");
+ return;
+ }
+ } else {
+ val = NULL;
+ }
+
+ encode_mod_path(mod, vnd, "sub", path, sizeof(path));
+
+ BT_DBG("Saving %s as %s", path, val ? val : "(null)");
+ settings_save_one(path, val);
+}
+
+static void store_pending_mod_pub(struct bt_mesh_model *mod, bool vnd)
+{
+ char buf[BT_SETTINGS_SIZE(sizeof(struct mod_pub_val))];
+ struct mod_pub_val pub;
+ char path[20];
+ char *val;
+
+ if (!mod->pub || mod->pub->addr == BT_MESH_ADDR_UNASSIGNED) {
+ val = NULL;
+ } else {
+ pub.addr = mod->pub->addr;
+ pub.key = mod->pub->key;
+ pub.ttl = mod->pub->ttl;
+ pub.retransmit = mod->pub->retransmit;
+ pub.period = mod->pub->period;
+ pub.period_div = mod->pub->period_div;
+ pub.cred = mod->pub->cred;
+
+ val = settings_str_from_bytes(&pub, sizeof(pub),
+ buf, sizeof(buf));
+ if (!val) {
+ BT_ERR("Unable to encode model publication as value");
+ return;
+ }
+ }
+
+ encode_mod_path(mod, vnd, "pub", path, sizeof(path));
+
+ BT_DBG("Saving %s as %s", path, val ? val : "(null)");
+ settings_save_one(path, val);
+}
+
+static void store_pending_mod(struct bt_mesh_model *mod,
+ struct bt_mesh_elem *elem, bool vnd,
+ bool primary, void *user_data)
+{
+ if (!mod->flags) {
+ return;
+ }
+
+ if (mod->flags & BT_MESH_MOD_BIND_PENDING) {
+ mod->flags &= ~BT_MESH_MOD_BIND_PENDING;
+ store_pending_mod_bind(mod, vnd);
+ }
+
+ if (mod->flags & BT_MESH_MOD_SUB_PENDING) {
+ mod->flags &= ~BT_MESH_MOD_SUB_PENDING;
+ store_pending_mod_sub(mod, vnd);
+ }
+
+ if (mod->flags & BT_MESH_MOD_PUB_PENDING) {
+ mod->flags &= ~BT_MESH_MOD_PUB_PENDING;
+ store_pending_mod_pub(mod, vnd);
+ }
+}
+
+static void store_pending(struct k_work *work)
+{
+ BT_DBG("");
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_RPL_PENDING)) {
+ if (bt_mesh.valid) {
+ store_pending_rpl();
+ } else {
+ clear_rpl();
+ }
+ }
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_KEYS_PENDING)) {
+ store_pending_keys();
+ }
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_NET_PENDING)) {
+ if (bt_mesh.valid) {
+ store_pending_net();
+ } else {
+ clear_net();
+ }
+ }
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_IV_PENDING)) {
+ if (bt_mesh.valid) {
+ store_pending_iv();
+ } else {
+ clear_iv();
+ }
+ }
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_SEQ_PENDING)) {
+ store_pending_seq();
+ }
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_HB_PUB_PENDING)) {
+ store_pending_hb_pub();
+ }
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_CFG_PENDING)) {
+ if (bt_mesh.valid) {
+ store_pending_cfg();
+ } else {
+ clear_cfg();
+ }
+ }
+
+ if (atomic_test_and_clear_bit(bt_mesh.flags, BT_MESH_MOD_PENDING)) {
+ bt_mesh_model_foreach(store_pending_mod, NULL);
+ }
+}
+
+void bt_mesh_store_rpl(struct bt_mesh_rpl *entry)
+{
+ entry->store = true;
+ schedule_store(BT_MESH_RPL_PENDING);
+}
+
+static struct key_update *key_update_find(bool app_key, u16_t key_idx,
+ struct key_update **free_slot)
+{
+ struct key_update *match;
+ int i;
+
+ match = NULL;
+ *free_slot = NULL;
+
+ for (i = 0; i < ARRAY_SIZE(key_updates); i++) {
+ struct key_update *update = &key_updates[i];
+
+ if (!update->valid) {
+ *free_slot = update;
+ continue;
+ }
+
+ if (update->app_key != app_key) {
+ continue;
+ }
+
+ if (update->key_idx == key_idx) {
+ match = update;
+ }
+ }
+
+ return match;
+}
+
+void bt_mesh_store_subnet(struct bt_mesh_subnet *sub)
+{
+ struct key_update *update, *free_slot;
+
+ BT_DBG("NetKeyIndex 0x%03x", sub->net_idx);
+
+ update = key_update_find(false, sub->net_idx, &free_slot);
+ if (update) {
+ update->clear = 0;
+ schedule_store(BT_MESH_KEYS_PENDING);
+ return;
+ }
+
+ if (!free_slot) {
+ store_net_key(sub);
+ return;
+ }
+
+ free_slot->valid = 1;
+ free_slot->key_idx = sub->net_idx;
+ free_slot->app_key = 0;
+ free_slot->clear = 0;
+
+ schedule_store(BT_MESH_KEYS_PENDING);
+}
+
+void bt_mesh_store_app_key(struct bt_mesh_app_key *key)
+{
+ struct key_update *update, *free_slot;
+
+ BT_DBG("AppKeyIndex 0x%03x", key->app_idx);
+
+ update = key_update_find(true, key->app_idx, &free_slot);
+ if (update) {
+ update->clear = 0;
+ schedule_store(BT_MESH_KEYS_PENDING);
+ return;
+ }
+
+ if (!free_slot) {
+ store_app_key(key);
+ return;
+ }
+
+ free_slot->valid = 1;
+ free_slot->key_idx = key->app_idx;
+ free_slot->app_key = 1;
+ free_slot->clear = 0;
+
+ schedule_store(BT_MESH_KEYS_PENDING);
+}
+
+void bt_mesh_store_hb_pub(void)
+{
+ schedule_store(BT_MESH_HB_PUB_PENDING);
+}
+
+void bt_mesh_store_cfg(void)
+{
+ schedule_store(BT_MESH_CFG_PENDING);
+}
+
+void bt_mesh_clear_net(void)
+{
+ schedule_store(BT_MESH_NET_PENDING);
+ schedule_store(BT_MESH_IV_PENDING);
+ schedule_store(BT_MESH_CFG_PENDING);
+}
+
+void bt_mesh_clear_subnet(struct bt_mesh_subnet *sub)
+{
+ struct key_update *update, *free_slot;
+
+ BT_DBG("NetKeyIndex 0x%03x", sub->net_idx);
+
+ update = key_update_find(false, sub->net_idx, &free_slot);
+ if (update) {
+ update->clear = 1;
+ schedule_store(BT_MESH_KEYS_PENDING);
+ return;
+ }
+
+ if (!free_slot) {
+ clear_net_key(sub->net_idx);
+ return;
+ }
+
+ free_slot->valid = 1;
+ free_slot->key_idx = sub->net_idx;
+ free_slot->app_key = 0;
+ free_slot->clear = 1;
+
+ schedule_store(BT_MESH_KEYS_PENDING);
+}
+
+void bt_mesh_clear_app_key(struct bt_mesh_app_key *key)
+{
+ struct key_update *update, *free_slot;
+
+ BT_DBG("AppKeyIndex 0x%03x", key->app_idx);
+
+ update = key_update_find(true, key->app_idx, &free_slot);
+ if (update) {
+ update->clear = 1;
+ schedule_store(BT_MESH_KEYS_PENDING);
+ return;
+ }
+
+ if (!free_slot) {
+ clear_app_key(key->app_idx);
+ return;
+ }
+
+ free_slot->valid = 1;
+ free_slot->key_idx = key->app_idx;
+ free_slot->app_key = 1;
+ free_slot->clear = 1;
+
+ schedule_store(BT_MESH_KEYS_PENDING);
+}
+
+void bt_mesh_clear_rpl(void)
+{
+ schedule_store(BT_MESH_RPL_PENDING);
+}
+
+void bt_mesh_store_mod_bind(struct bt_mesh_model *mod)
+{
+ mod->flags |= BT_MESH_MOD_BIND_PENDING;
+ schedule_store(BT_MESH_MOD_PENDING);
+}
+
+void bt_mesh_store_mod_sub(struct bt_mesh_model *mod)
+{
+ mod->flags |= BT_MESH_MOD_SUB_PENDING;
+ schedule_store(BT_MESH_MOD_PENDING);
+}
+
+void bt_mesh_store_mod_pub(struct bt_mesh_model *mod)
+{
+ mod->flags |= BT_MESH_MOD_PUB_PENDING;
+ schedule_store(BT_MESH_MOD_PENDING);
+}
+
+int bt_mesh_settings_init(void)
+{
+ k_delayed_work_init(&pending_store, store_pending);
+
+ int err;
+
+ BT_DBG("");
+
+ err = settings_subsys_init();
+ if (err) {
+ BT_ERR("settings_subsys_init failed (err %d)", err);
+ return err;
+ }
+
+ err = settings_register(&mesh);
+ if (err) {
+ BT_ERR("settings_register failed (err %d)", err);
+ return err;
+ }
+
+ err = settings_load();
+ if (err) {
+ BT_ERR("settings load failed (err %d)", err);
+ return err;
+ }
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_SETTINGS */
diff --git a/components/bt/ble_mesh/mesh_core/settings/include/fcb.h b/components/bt/ble_mesh/mesh_core/settings/include/fcb.h
new file mode 100644
index 0000000000..a016d1dce9
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/include/fcb.h
@@ -0,0 +1,144 @@
+/*
+ * Copyright (c) 2017 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef __FCB_H_
+#define __FCB_H_
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/*
+ * Flash circular buffer.
+ */
+#include
+#include
+#include "mesh_types.h"
+
+
+#define FCB_MAX_LEN (CHAR_MAX | CHAR_MAX << 7) /* Max length of element */
+
+/*
+ * Entry location is pointer to area (within fcb->f_sectors), and offset
+ * within that area.
+ */
+struct fcb_entry {
+ struct flash_sector *fe_sector; /* ptr to sector */
+ /* within fcb->f_sectors */
+ u32_t fe_elem_off; /* start of entry */
+ u32_t fe_data_off; /* start of data */
+ u16_t fe_data_len; /* size of data area */
+};
+
+/*
+ * Helper macro for calculate the data offset related to
+ * the fcb flash_area start offset.
+ */
+#define FCB_ENTRY_FA_DATA_OFF(entry) (entry.fe_sector->fs_off +\
+ entry.fe_data_off)
+
+struct fcb_entry_ctx {
+ struct fcb_entry loc;
+ const struct flash_area *fap;
+};
+
+struct fcb {
+ /* Caller of fcb_init fills this in */
+ u32_t f_magic; /* As placed on the disk */
+ u8_t f_version; /* Current version number of the data */
+ u8_t f_sector_cnt; /* Number of elements in sector array */
+ u8_t f_scratch_cnt; /* How many sectors should be kept empty */
+ struct flash_sector *f_sectors; /* Array of sectors, */
+ /* must be contiguous */
+
+ /* Flash circular buffer internal state */
+ //struct k_mutex f_mtx; /* Locking for accessing the FCB data */
+ struct flash_sector *f_oldest;
+ struct fcb_entry f_active;
+ u16_t f_active_id;
+ u8_t f_align; /* writes to flash have to aligned to this */
+
+ const struct flash_area *fap; /* Flash area used by the fcb instance */
+ /* This can be transfer to FCB user */
+};
+
+/*
+ * Error codes.
+ */
+#define FCB_OK 0
+#define FCB_ERR_ARGS -1
+#define FCB_ERR_FLASH -2
+#define FCB_ERR_NOVAR -3
+#define FCB_ERR_NOSPACE -4
+#define FCB_ERR_NOMEM -5
+#define FCB_ERR_CRC -6
+#define FCB_ERR_MAGIC -7
+
+int fcb_init(int f_area_id, struct fcb *fcb);
+
+/*
+ * fcb_append() appends an entry to circular buffer. When writing the
+ * contents for the entry, use loc->fl_area and loc->fl_data_off with
+ * flash_area_write(). When you're finished, call fcb_append_finish() with
+ * loc as argument.
+ */
+int fcb_append(struct fcb *fcb, u16_t len, struct fcb_entry *loc);
+int fcb_append_finish(struct fcb *fcb, struct fcb_entry *append_loc);
+
+/*
+ * Walk over all log entries in FCB, or entries in a given flash_area.
+ * cb gets called for every entry. If cb wants to stop the walk, it should
+ * return non-zero value.
+ *
+ * Entry data can be read using flash_area_read(), using
+ * loc->fe_area, loc->fe_data_off, and loc->fe_data_len as arguments.
+ */
+typedef int (*fcb_walk_cb)(struct fcb_entry_ctx *loc_ctx, void *arg);
+int fcb_walk(struct fcb *fcb, struct flash_sector *sector, fcb_walk_cb cb,
+ void *cb_arg);
+int fcb_getnext(struct fcb *fcb, struct fcb_entry *loc);
+
+/*
+ * Erases the data from oldest sector.
+ */
+int fcb_rotate(struct fcb *fcb);
+
+/*
+ * Start using the scratch block.
+ */
+int fcb_append_to_scratch(struct fcb *fcb);
+
+/*
+ * How many sectors are unused.
+ */
+int fcb_free_sector_cnt(struct fcb *fcb);
+
+/*
+ * Whether FCB has any data.
+ */
+int fcb_is_empty(struct fcb *fcb);
+
+/*
+ * Element at offset *entries* from last position (backwards).
+ */
+int fcb_offset_last_n(struct fcb *fcb, u8_t entries,
+ struct fcb_entry *last_n_entry);
+
+/*
+ * Clears FCB passed to it
+ */
+int fcb_clear(struct fcb *fcb);
+
+int fcb_flash_read(const struct fcb *fcb, const struct flash_sector *sector,
+ off_t off, void *dst, size_t len);
+int fcb_flash_write(const struct fcb *fcb, const struct flash_sector *sector,
+ off_t off, const void *src, size_t len);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __FCB_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/settings/include/settings.h b/components/bt/ble_mesh/mesh_core/settings/include/settings.h
new file mode 100644
index 0000000000..5ffe6bff7b
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/include/settings.h
@@ -0,0 +1,271 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __SETTINGS_H_
+#define __SETTINGS_H_
+
+#include
+
+#include "mesh_util.h"
+#include "mesh_slist.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @defgroup settings Settings subsystem
+ * @{
+ */
+
+#define SETTINGS_MAX_DIR_DEPTH 8 /* max depth of settings tree */
+#define SETTINGS_MAX_NAME_LEN (8 * SETTINGS_MAX_DIR_DEPTH)
+#define SETTINGS_MAX_VAL_LEN 256
+#define SETTINGS_NAME_SEPARATOR "/"
+
+/* pleace for settings additions:
+ * up to 7 separators, '=', '\0'
+ */
+#define SETTINGS_EXTRA_LEN ((SETTINGS_MAX_DIR_DEPTH - 1) + 2)
+
+#define SETTINGS_NMGR_OP 0
+
+/**
+ * Type of settings value.
+ */
+enum settings_type {
+ SETTINGS_NONE = 0,
+ SETTINGS_INT8,
+ SETTINGS_INT16,
+ SETTINGS_INT32,
+ SETTINGS_INT64,
+ SETTINGS_STRING,
+ SETTINGS_BYTES,
+ SETTINGS_FLOAT,
+ SETTINGS_DOUBLE,
+ SETTINGS_BOOL,
+} __attribute__((__packed__));
+
+/**
+ * Parameter to commit handler describing where data is going to.
+ */
+enum settings_export_tgt {
+ SETTINGS_EXPORT_PERSIST, /* Value is to be persisted. */
+ SETTINGS_EXPORT_SHOW /* Value is to be displayed. */
+};
+
+/**
+ * @struct settings_handler
+ * Config handlers for subtree implement a set of handler functions.
+ * These are registered using a call to settings_register().
+ *
+ * @param settings_handler::node Linked list node info for module internal usage.
+ *
+ * @param settings_handler::name Name of subtree.
+ *
+ * @param settings_handler::h_get Get values handler of settings items
+ * identified by keyword names.Parameters:
+ * - argc - count of item in argv.
+ * - argv - array of pointers to keyword names.
+ * - val - buffer for a value.
+ * - val_len_max - size of that buffer.
+ *
+ * @param settings_handler::h_set Sey value handler of settings items
+ * identified by keyword names. Parameters:
+ * - argc - count of item in argv, argv - array of pointers to keyword names.
+ * - val- pointer to value to be set.
+ *
+ * @param settings_handler::h_commit This handler gets called after settings
+ * has been loaded in full. User might use it to apply setting to
+ * the application.
+ *
+ * @param settings_handler::h_export This gets called to dump all current
+ * settings items.
+ * This happens when settings_save() tries to save the settings. Parameters:
+ * - tgt: indicates where data is going to.
+ * - Export_function: the pointer to the internal function which appends
+ * a single key-value pair to persisted settings. Don't store duplicated
+ * value. The name is subtree/key string, val is the string with
+ * value.
+ *
+ * @remarks The User might limit a implementations of handler to serving only
+ * one keyword at one call - what will impose limit to get/set values using full
+ * subtree/key name.
+ */
+struct settings_handler {
+ sys_snode_t node;
+ char *name;
+ char *(*h_get)(int argc, char **argv, char *val, int val_len_max);
+ int (*h_set)(int argc, char **argv, char *val);
+ int (*h_commit)(void);
+ int (*h_export)(int (*export_func)(const char *name, char *val),
+ enum settings_export_tgt tgt);
+};
+
+/**
+ * Initialization of settings and backend
+ *
+ * Can be called at application startup.
+ * In case the backend is NFFS Remember to call it after FS was mounted.
+ * For FCB backend it can be called without such a restriction.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_subsys_init(void);
+
+/**
+ * Register a handler for settings items.
+ *
+ * @param cf Structure containing registration info.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_register(struct settings_handler *cf);
+
+/**
+ * Load serialized items from registered persistence sources. Handlers for
+ * serialized item subtrees registered earlier will be called for encountered
+ * values.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_load(void);
+
+/**
+ * Save currently running serialized items. All serialized items which are different
+ * from currently persisted values will be saved.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_save(void);
+
+/**
+ * Write a single serialized value to persisted storage (if it has
+ * changed value).
+ *
+ * @param name Name/key of the settings item.
+ * @param var Value of the settings item.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_save_one(const char *name, char *var);
+
+/**
+ * Set settings item identified by @p name to be value @p val_str.
+ * This finds the settings handler for this subtree and calls it's
+ * set handler.
+ *
+ * @param name Name/key of the settings item.
+ * @param val_str Value of the settings item.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_set_value(char *name, char *val_str);
+
+/**
+ * Get value of settings item identified by @p name.
+ * This calls the settings handler h_get for the subtree.
+ *
+ * Configuration handler can copy the string to @p buf, the maximum
+ * number of bytes it will copy is limited by @p buf_len.
+ *
+ * @param name Name/key of the settings item.
+ *
+ * @param buf buffer for value of the settings item.
+ * If value is not string, the value will be filled in *buf.
+ *
+ * @param buf_len size of buf.
+ *
+ * @return value will be pointer to beginning of the buf,
+ * except for string it will pointer to beginning of string source.
+ */
+char *settings_get_value(char *name, char *buf, int buf_len);
+
+/**
+ * Call commit for all settings handler. This should apply all
+ * settings which has been set, but not applied yet.
+ *
+ * @param name Name of the settings subtree, or NULL to commit everything.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_commit(char *name);
+
+/**
+ * Convenience routine for converting value passed as a string to native
+ * data type.
+ *
+ * @param val_str Value of the settings item as string.
+ * @param type Type of the value to convert to.
+ * @param vp Pointer to variable to fill with the decoded value.
+ * @param maxlen the vp buffer size.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_value_from_str(char *val_str, enum settings_type type, void *vp,
+ int maxlen);
+
+/**
+ * Convenience routine for converting byte array passed as a base64
+ * encoded string.
+ *
+ * @param val_str Value of the settings item as string.
+ * @param vp Pointer to variable to fill with the decoded value.
+ * @param len Size of that variable. On return the number of bytes in the array.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+int settings_bytes_from_str(char *val_str, void *vp, int *len);
+
+/**
+ * Convenience routine for converting native data type to a string.
+ *
+ * @param type Type of the value to convert from.
+ * @param vp Pointer to variable to convert.
+ * @param buf Buffer where string value will be stored.
+ * @param buf_len Size of the buffer.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+char *settings_str_from_value(enum settings_type type, void *vp, char *buf,
+ int buf_len);
+#define SETTINGS_STR_FROM_BYTES_LEN(len) (((len) * 4 / 3) + 4)
+
+/**
+ * Convenience routine for converting byte array into a base64
+ * encoded string.
+ *
+ * @param vp Pointer to variable to convert.
+ * @param vp_len Number of bytes to convert.
+ * @param buf Buffer where string value will be stored.
+ * @param buf_len Size of the buffer.
+ *
+ * @return 0 on success, non-zero on failure.
+ */
+char *settings_str_from_bytes(void *vp, int vp_len, char *buf, int buf_len);
+
+#define SETTINGS_VALUE_SET(str, type, val) \
+ settings_value_from_str((str), (type), &(val), sizeof(val))
+
+/**
+ * @} settings
+ */
+
+/*
+ * Config storage
+ */
+struct settings_store_itf;
+struct settings_store {
+ sys_snode_t cs_next;
+ const struct settings_store_itf *cs_itf;
+};
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __SETTINGS_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/settings/include/settings_fcb.h b/components/bt/ble_mesh/mesh_core/settings/include/settings_fcb.h
new file mode 100644
index 0000000000..2ae1228288
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/include/settings_fcb.h
@@ -0,0 +1,39 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __SETTINGS_FCB_H_
+#define __SETTINGS_FCB_H_
+
+#include "nvs.h"
+#include "include/settings.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+struct settings_fcb {
+ struct settings_store cf_store;
+ const char *file_name;
+ nvs_handle handle;
+};
+
+struct settings_dup_check_arg {
+ const char *name;
+ const char *val;
+ int is_dup;
+};
+
+extern int settings_fcb_src(struct settings_fcb *cf);
+extern int settings_fcb_dst(struct settings_fcb *cf);
+
+int csi_load_check(struct settings_dup_check_arg *cdca);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __SETTINGS_FCB_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/settings/include/settings_file.h b/components/bt/ble_mesh/mesh_core/settings/include/settings_file.h
new file mode 100644
index 0000000000..a41ca143fa
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/include/settings_file.h
@@ -0,0 +1,36 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __SETTINGS_FILE_H_
+#define __SETTINGS_FILE_H_
+
+#include "settings/settings.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#define SETTINGS_FILE_NAME_MAX 32 /* max length for settings filename */
+
+struct settings_file {
+ struct settings_store cf_store;
+ const char *cf_name; /* filename */
+ int cf_maxlines; /* max # of lines before compressing */
+ int cf_lines; /* private */
+};
+
+/* register file to be source of settings */
+int settings_file_src(struct settings_file *cf);
+
+/* settings saves go to a file */
+int settings_file_dst(struct settings_file *cf);
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __SETTINGS_FILE_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/settings/include/settings_priv.h b/components/bt/ble_mesh/mesh_core/settings/include/settings_priv.h
new file mode 100644
index 0000000000..875066d4aa
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/include/settings_priv.h
@@ -0,0 +1,50 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef __SETTINGS_PRIV_H_
+#define __SETTINGS_PRIV_H_
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+int settings_cli_register(void);
+int settings_nmgr_register(void);
+
+struct mgmt_cbuf;
+int settings_cbor_line(struct mgmt_cbuf *cb, char *name, int nlen, char *value,
+ int vlen);
+
+int settings_line_parse(char *buf, char **namep, char **valp);
+int settings_line_make(char *dst, int dlen, const char *name, const char *val);
+int settings_line_make2(char *dst, int dlen, const char *name,
+ const char *value);
+
+/*
+ * API for config storage.
+ */
+typedef void (*load_cb)(char *name, char *val, void *cb_arg);
+struct settings_store_itf {
+ int (*csi_load)(struct settings_store *cs, load_cb cb, void *cb_arg);
+ int (*csi_save_start)(struct settings_store *cs);
+ int (*csi_save)(struct settings_store *cs, const char *name,
+ const char *value);
+ int (*csi_save_end)(struct settings_store *cs);
+};
+
+void settings_src_register(struct settings_store *cs);
+void settings_dst_register(struct settings_store *cs);
+
+extern sys_slist_t settings_load_srcs;
+extern sys_slist_t settings_handlers;
+extern struct settings_store *settings_save_dst;
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif /* __SETTINGS_PRIV_H_ */
diff --git a/components/bt/ble_mesh/mesh_core/settings/settings.c b/components/bt/ble_mesh/mesh_core/settings/settings.c
new file mode 100644
index 0000000000..382b55d235
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/settings.c
@@ -0,0 +1,396 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#include
+#include
+#include
+#include
+#include
+
+#include "sdkconfig.h"
+
+#include "mesh_types.h"
+#include "base64.h"
+#include "settings/include/settings.h"
+#include "settings/include/settings_priv.h"
+
+#if CONFIG_BT_MESH_SETTINGS
+
+/* mbedtls-base64 lib encodes data to null-terminated string */
+#define BASE64_ENCODE_SIZE(in_size) ((((((in_size) - 1) / 3) * 4) + 4) + 1)
+
+sys_slist_t settings_handlers;
+
+static u8_t settings_cmd_inited;
+
+void settings_store_init(void);
+static void s64_to_dec(char *ptr, int buf_len, s64_t value, int base);
+static s64_t dec_to_s64(char *p_str, char **e_ptr);
+
+void settings_init(void)
+{
+ if (!settings_cmd_inited) {
+ sys_slist_init(&settings_handlers);
+ settings_store_init();
+
+ settings_cmd_inited = 1;
+ }
+}
+
+int settings_register(struct settings_handler *handler)
+{
+ sys_slist_prepend(&settings_handlers, &handler->node);
+ return 0;
+}
+
+/*
+ * Find settings_handler based on name.
+ */
+struct settings_handler *settings_handler_lookup(char *name)
+{
+ struct settings_handler *ch;
+
+ SYS_SLIST_FOR_EACH_CONTAINER(&settings_handlers, ch, node) {
+ BT_DBG("%s, name = %s, ch->name = %s", __func__, name, ch->name);
+ if (!strcmp(name, ch->name)) {
+ return ch;
+ }
+ }
+ return NULL;
+}
+
+/*
+ * Separate string into argv array.
+ */
+int settings_parse_name(char *name, int *name_argc, char *name_argv[])
+{
+ int i = 0;
+
+ while (name) {
+ name_argv[i++] = name;
+
+ while (1) {
+ if (*name == '\0') {
+ name = NULL;
+ break;
+ }
+
+ if (*name == *SETTINGS_NAME_SEPARATOR) {
+ *name = '\0';
+ name++;
+ break;
+ }
+ name++;
+ }
+ }
+
+ *name_argc = i;
+
+ return 0;
+}
+
+static struct settings_handler *settings_parse_and_lookup(char *name,
+ int *name_argc,
+ char *name_argv[])
+{
+ int rc;
+
+ rc = settings_parse_name(name, name_argc, name_argv);
+ if (rc) {
+ BT_ERR("settings_parse_name fail");
+ return NULL;
+ }
+ return settings_handler_lookup(name_argv[0]);
+}
+
+int settings_value_from_str(char *val_str, enum settings_type type, void *vp,
+ int maxlen)
+{
+ s32_t val;
+ s64_t val64;
+ char *eptr;
+
+ if (!val_str) {
+ goto err;
+ }
+ switch (type) {
+ case SETTINGS_INT8:
+ case SETTINGS_INT16:
+ case SETTINGS_INT32:
+ case SETTINGS_BOOL:
+ val = strtol(val_str, &eptr, 0);
+ if (*eptr != '\0') {
+ goto err;
+ }
+ if (type == SETTINGS_BOOL) {
+ if (val < 0 || val > 1) {
+ goto err;
+ }
+ *(bool *)vp = val;
+ } else if (type == SETTINGS_INT8) {
+ if (val < INT8_MIN || val > UINT8_MAX) {
+ goto err;
+ }
+ *(int8_t *)vp = val;
+ } else if (type == SETTINGS_INT16) {
+ if (val < INT16_MIN || val > UINT16_MAX) {
+ goto err;
+ }
+ *(int16_t *)vp = val;
+ } else if (type == SETTINGS_INT32) {
+ *(s32_t *)vp = val;
+ }
+ break;
+ case SETTINGS_INT64:
+ val64 = dec_to_s64(val_str, &eptr);
+ if (*eptr != '\0') {
+ goto err;
+ }
+ *(s64_t *)vp = val64;
+ break;
+ case SETTINGS_STRING:
+ val = strlen(val_str);
+ if (val + 1 > maxlen) {
+ goto err;
+ }
+ strcpy(vp, val_str);
+ break;
+ default:
+ goto err;
+ }
+ return 0;
+err:
+ return -EINVAL;
+}
+
+int settings_bytes_from_str(char *val_str, void *vp, int *len)
+{
+ int err;
+ size_t rc;
+
+ err = base64_decode(vp, *len, &rc, (const u8_t *)val_str, strlen(val_str));
+
+ if (err) {
+ return -1;
+ }
+
+ *len = rc;
+ return 0;
+}
+
+char *settings_str_from_value(enum settings_type type, void *vp, char *buf,
+ int buf_len)
+{
+ s32_t val;
+
+ if (type == SETTINGS_STRING) {
+ return vp;
+ }
+ switch (type) {
+ case SETTINGS_INT8:
+ case SETTINGS_INT16:
+ case SETTINGS_INT32:
+ case SETTINGS_BOOL:
+ if (type == SETTINGS_BOOL) {
+ val = *(bool *)vp;
+ } else if (type == SETTINGS_INT8) {
+ val = *(int8_t *)vp;
+ } else if (type == SETTINGS_INT16) {
+ val = *(int16_t *)vp;
+ } else {
+ val = *(s32_t *)vp;
+ }
+ snprintf(buf, buf_len, "%ld", (long)val);
+ return buf;
+ case SETTINGS_INT64:
+ s64_to_dec(buf, buf_len, *(s64_t *)vp, 10);
+ return buf;
+ default:
+ return NULL;
+ }
+}
+
+static void u64_to_dec(char *ptr, int buf_len, u64_t value, int base)
+{
+ u64_t t = 0, res = 0;
+ u64_t tmp = value;
+ int count = 0;
+
+ if (ptr == NULL) {
+ return;
+ }
+
+ if (tmp == 0) {
+ count++;
+ }
+
+ while (tmp > 0) {
+ tmp = tmp / base;
+ count++;
+ }
+
+ ptr += count;
+
+ *ptr = '\0';
+
+ do {
+ res = value - base * (t = value / base);
+ if (res < 10) {
+ *--ptr = '0' + res;
+ } else if ((res >= 10) && (res < 16)) {
+ *--ptr = 'A' - 10 + res;
+ }
+ value = t;
+ } while (value != 0);
+}
+
+static void s64_to_dec(char *ptr, int buf_len, s64_t value, int base)
+{
+ u64_t val64;
+
+ if (ptr == NULL || buf_len < 1) {
+ return;
+ }
+
+ if (value < 0) {
+ *ptr = '-';
+ ptr++;
+ buf_len--;
+ val64 = value * (-1);
+ } else {
+ val64 = value;
+ }
+
+ u64_to_dec(ptr, buf_len, val64, base);
+}
+
+static s64_t dec_to_s64(char *p_str, char **e_ptr)
+{
+ u64_t val = 0, prev_val = 0;
+ bool neg = false;
+ int digit;
+
+ if (*p_str == '-') {
+ neg = true;
+ p_str++;
+ } else if (*p_str == '+') {
+ p_str++;
+ }
+
+ while (1) {
+ if (*p_str >= '0' && *p_str <= '9') {
+ digit = *p_str - '0';
+ } else {
+ break;
+ }
+
+ val *= 10;
+ val += digit;
+
+ /* this is only a fuse */
+ if (val < prev_val) {
+ break;
+ }
+
+ prev_val = val;
+ p_str++;
+ }
+
+ if (e_ptr != 0) {
+ *e_ptr = p_str;
+ }
+
+ if (neg) {
+ return -val;
+ } else {
+ return val;
+ }
+}
+
+char *settings_str_from_bytes(void *vp, int vp_len, char *buf, int buf_len)
+{
+ if (BASE64_ENCODE_SIZE(vp_len) > buf_len) {
+ return NULL;
+ }
+
+ size_t enc_len;
+
+ base64_encode((u8_t *)buf, buf_len, &enc_len, vp, vp_len);
+
+ return buf;
+}
+
+int settings_set_value(char *name, char *val_str)
+{
+ int name_argc;
+ char *name_argv[SETTINGS_MAX_DIR_DEPTH];
+ struct settings_handler *ch;
+
+ ch = settings_parse_and_lookup(name, &name_argc, name_argv);
+ if (!ch) {
+ BT_ERR("settings_parse_and_lookup fail.");
+ return -EINVAL;
+ }
+
+ return ch->h_set(name_argc - 1, &name_argv[1], val_str);
+}
+
+/*
+ * Get value in printable string form. If value is not string, the value
+ * will be filled in *buf.
+ * Return value will be pointer to beginning of that buffer,
+ * except for string it will pointer to beginning of string.
+ */
+char *settings_get_value(char *name, char *buf, int buf_len)
+{
+ int name_argc;
+ char *name_argv[SETTINGS_MAX_DIR_DEPTH];
+ struct settings_handler *ch;
+
+ ch = settings_parse_and_lookup(name, &name_argc, name_argv);
+ if (!ch) {
+ return NULL;
+ }
+
+ if (!ch->h_get) {
+ return NULL;
+ }
+ return ch->h_get(name_argc - 1, &name_argv[1], buf, buf_len);
+}
+
+int settings_commit(char *name)
+{
+ int name_argc;
+ char *name_argv[SETTINGS_MAX_DIR_DEPTH];
+ struct settings_handler *ch;
+ int rc;
+ int rc2;
+
+ if (name) {
+ ch = settings_parse_and_lookup(name, &name_argc, name_argv);
+ if (!ch) {
+ return -EINVAL;
+ }
+ if (ch->h_commit) {
+ return ch->h_commit();
+ } else {
+ return 0;
+ }
+ } else {
+ rc = 0;
+ SYS_SLIST_FOR_EACH_CONTAINER(&settings_handlers, ch, node) {
+ if (ch->h_commit) {
+ rc2 = ch->h_commit();
+ if (!rc) {
+ rc = rc2;
+ }
+ }
+ }
+ return rc;
+ }
+}
+
+#endif /* CONFIG_BT_MESH_SETTINGS */
diff --git a/components/bt/ble_mesh/mesh_core/settings/settings_fcb.c b/components/bt/ble_mesh/mesh_core/settings/settings_fcb.c
new file mode 100644
index 0000000000..0f965f5857
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/settings_fcb.c
@@ -0,0 +1,198 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+
+#include "sdkconfig.h"
+#include "nvs.h"
+
+#include "settings/include/settings.h"
+#include "settings/include/settings_fcb.h"
+#include "settings/include/settings_priv.h"
+
+#if CONFIG_BT_MESH_SETTINGS
+
+#define SETTINGS_FCB_VERS 1
+#define KEY_BUF_LEN_MAX 1024
+
+static int buf_len = 0;
+static const char *BLE_MESH_FCB_NVS_KEY = "fcb_nvs_key";
+static const char *BLE_MESH_FCB_FILE_NAME = "fcb_nvs_file";
+static char key_buf[KEY_BUF_LEN_MAX];
+
+struct settings_fcb_load_cb_arg {
+ load_cb cb;
+ void *cb_arg;
+};
+
+static int settings_fcb_load(struct settings_store *cs, load_cb cb,
+ void *cb_arg);
+static int settings_fcb_save(struct settings_store *cs, const char *name,
+ const char *value);
+
+static struct settings_store_itf settings_fcb_itf = {
+ .csi_load = settings_fcb_load,
+ .csi_save = settings_fcb_save,
+};
+
+int settings_fcb_src(struct settings_fcb *cf)
+{
+ cf->file_name = BLE_MESH_FCB_FILE_NAME;
+ if (nvs_open(BLE_MESH_FCB_FILE_NAME, NVS_READWRITE, &cf->handle) != ESP_OK) {
+ return -1;
+ }
+ cf->cf_store.cs_itf = &settings_fcb_itf;
+ settings_src_register(&cf->cf_store);
+
+ return 0;
+}
+
+int settings_fcb_dst(struct settings_fcb *cf)
+{
+ cf->cf_store.cs_itf = &settings_fcb_itf;
+ settings_dst_register(&cf->cf_store);
+
+ return 0;
+}
+
+static int settings_fcb_load_cb(struct settings_fcb *cf, void *arg)
+{
+ struct settings_fcb_load_cb_arg *argp;
+ char buf[SETTINGS_MAX_NAME_LEN + SETTINGS_MAX_VAL_LEN +
+ SETTINGS_EXTRA_LEN];
+ char *name_str;
+ char *val_str;
+ int rc = 0;
+ size_t len = KEY_BUF_LEN_MAX;
+ char *cp_start = NULL;
+
+ argp = (struct settings_fcb_load_cb_arg *)arg;
+
+ rc = nvs_get_blob(cf->handle, BLE_MESH_FCB_NVS_KEY, NULL, (size_t *)&buf_len);
+ rc = nvs_get_blob(cf->handle, BLE_MESH_FCB_NVS_KEY, key_buf, (size_t *)&buf_len);
+
+ if (rc) {
+ return 0;
+ }
+
+ cp_start = key_buf;
+ while (cp_start != NULL && *cp_start != '\0') {
+ len = strlen(cp_start);
+ if (len >= sizeof(buf)) {
+ len = sizeof(buf) - 1;
+ }
+
+ memcpy(buf, cp_start, len);
+ buf[len] = '\0';
+ rc = settings_line_parse(buf, &name_str, &val_str);
+
+ if (rc) {
+ return 0;
+ }
+ BT_DBG("name_str = %s, val_str = %s", name_str, val_str);
+ argp->cb(name_str, val_str, argp->cb_arg);
+ cp_start += (len + 1);
+ }
+ return 0;
+}
+
+static int settings_fcb_load(struct settings_store *cs, load_cb cb,
+ void *cb_arg)
+{
+ struct settings_fcb *cf = (struct settings_fcb *)cs;
+ struct settings_fcb_load_cb_arg arg;
+ int rc;
+
+ arg.cb = cb;
+ arg.cb_arg = cb_arg;
+ rc = settings_fcb_load_cb(cf, &arg);
+ if (rc) {
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+static char *settings_link_buf(char *buf)
+{
+
+ int blen = strlen(buf);
+
+ memcpy(&key_buf[buf_len], buf, blen);
+ buf_len += blen;
+
+ key_buf[buf_len++] = '\0';
+
+ return key_buf;
+}
+
+static int settings_fcb_save(struct settings_store *cs, const char *name,
+ const char *value)
+{
+ struct settings_fcb *cf = (struct settings_fcb *)cs;
+ char buf[SETTINGS_MAX_NAME_LEN + SETTINGS_MAX_VAL_LEN +
+ SETTINGS_EXTRA_LEN];
+ int len;
+
+ if (!name) {
+ return -EINVAL;
+ }
+
+ len = settings_line_make(buf, sizeof(buf), name, value);
+ if (len < 0 || len + 2 > sizeof(buf)) {
+ return -EINVAL;
+ }
+
+ settings_link_buf(buf);
+ return nvs_set_blob(cf->handle, BLE_MESH_FCB_NVS_KEY, (void *)key_buf, buf_len);
+}
+
+int csi_load_check(struct settings_dup_check_arg *cdca)
+{
+ char *cp_start = NULL;
+ cp_start = key_buf;
+ size_t len = 0;
+ char buf[SETTINGS_MAX_NAME_LEN + SETTINGS_MAX_VAL_LEN +
+ SETTINGS_EXTRA_LEN];
+ char *name_str;
+ char *val_str;
+ int rc = 0;
+
+ while (cp_start != NULL && *cp_start != '\0') {
+ len = strlen(cp_start);
+ if (len >= sizeof(buf)) {
+ len = sizeof(buf) - 1;
+ }
+
+ memcpy(buf, cp_start, len);
+ buf[len] = '\0';
+ rc = settings_line_parse(buf, &name_str, &val_str);
+ if (rc) {
+ cp_start += (len + 1);
+ continue;
+ }
+
+ if (strcmp(name_str, cdca->name)) {
+ cp_start += (len + 1);
+ continue;
+ }
+
+ if (cdca->val && !strcmp(val_str, cdca->val)) {
+ cdca->is_dup = 1;
+ return 0;
+ } else {
+ cdca->is_dup = 0;
+ }
+
+ cp_start += (len + 1);
+ }
+
+ return 0;
+}
+
+#endif /* CONFIG_BT_MESH_SETTINGS */
diff --git a/components/bt/ble_mesh/mesh_core/settings/settings_file.c b/components/bt/ble_mesh/mesh_core/settings/settings_file.c
new file mode 100644
index 0000000000..db8e1027e1
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/settings_file.c
@@ -0,0 +1,303 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+
+#ifdef CONFIG_SETTINGS_FS
+
+#include "settings/include/settings.h"
+#include "settings/include/settings_file.h"
+#include "settings/include/settings_priv.h"
+
+static int settings_file_load(struct settings_store *cs, load_cb cb,
+ void *cb_arg);
+static int settings_file_save(struct settings_store *cs, const char *name,
+ const char *value);
+
+static struct settings_store_itf settings_file_itf = {
+ .csi_load = settings_file_load,
+ .csi_save = settings_file_save,
+};
+
+/*
+ * Register a file to be a source of configuration.
+ */
+int settings_file_src(struct settings_file *cf)
+{
+ if (!cf->cf_name) {
+ return -EINVAL;
+ }
+ cf->cf_store.cs_itf = &settings_file_itf;
+ settings_src_register(&cf->cf_store);
+
+ return 0;
+}
+
+int settings_file_dst(struct settings_file *cf)
+{
+ if (!cf->cf_name) {
+ return -EINVAL;
+ }
+ cf->cf_store.cs_itf = &settings_file_itf;
+ settings_dst_register(&cf->cf_store);
+
+ return 0;
+}
+
+int settings_getnext_line(struct fs_file_t *file, char *buf, int blen,
+ off_t *loc)
+{
+ int rc;
+ char *end;
+
+ rc = fs_seek(file, *loc, FS_SEEK_SET);
+ if (rc < 0) {
+ *loc = 0;
+ return -1;
+ }
+
+ rc = fs_read(file, buf, blen);
+ if (rc <= 0) {
+ *loc = 0;
+ return -1;
+ }
+
+ if (rc == blen) {
+ rc--;
+ }
+ buf[rc] = '\0';
+
+ end = strchr(buf, '\n');
+ if (end) {
+ *end = '\0';
+ } else {
+ end = strchr(buf, '\0');
+ }
+ blen = end - buf;
+ *loc += (blen + 1);
+ return blen;
+}
+
+/*
+ * Called to load configuration items. cb must be called for every configuration
+ * item found.
+ */
+static int settings_file_load(struct settings_store *cs, load_cb cb,
+ void *cb_arg)
+{
+ struct settings_file *cf = (struct settings_file *)cs;
+ struct fs_file_t file;
+ off_t loc;
+ char tmpbuf[SETTINGS_MAX_NAME_LEN + SETTINGS_MAX_VAL_LEN +
+ SETTINGS_EXTRA_LEN];
+ char *name_str;
+ char *val_str;
+ int rc;
+ int lines;
+
+ rc = fs_open(&file, cf->cf_name);
+ if (rc != 0) {
+ return -EINVAL;
+ }
+
+ loc = 0;
+ lines = 0;
+ while (1) {
+ rc = settings_getnext_line(&file, tmpbuf, sizeof(tmpbuf), &loc);
+ if (loc == 0) {
+ break;
+ }
+ if (rc < 0) {
+ continue;
+ }
+ rc = settings_line_parse(tmpbuf, &name_str, &val_str);
+ if (rc != 0) {
+ continue;
+ }
+ lines++;
+ cb(name_str, val_str, cb_arg);
+ }
+ rc = fs_close(&file);
+ cf->cf_lines = lines;
+
+ return rc;
+}
+
+static void settings_tmpfile(char *dst, const char *src, char *pfx)
+{
+ int len;
+ int pfx_len;
+
+ len = strlen(src);
+ pfx_len = strlen(pfx);
+ if (len + pfx_len >= SETTINGS_FILE_NAME_MAX) {
+ len = SETTINGS_FILE_NAME_MAX - pfx_len - 1;
+ }
+ memcpy(dst, src, len);
+ memcpy(dst + len, pfx, pfx_len);
+ dst[len + pfx_len] = '\0';
+}
+
+static int settings_file_create_or_replace(struct fs_file_t *zfp,
+ const char *file_name)
+{
+ struct fs_dirent entry;
+
+ if (fs_stat(file_name, &entry) == 0) {
+ if (entry.type == FS_DIR_ENTRY_FILE) {
+ if (fs_unlink(file_name)) {
+ return -EIO;
+ }
+ } else {
+ return -EISDIR;
+ }
+ }
+
+ return fs_open(zfp, file_name);
+}
+/*
+ * Try to compress configuration file by keeping unique names only.
+ */
+void settings_file_compress(struct settings_file *cf)
+{
+ int rc;
+ struct fs_file_t rf;
+ struct fs_file_t wf;
+ char tmp_file[SETTINGS_FILE_NAME_MAX];
+ char buf1[SETTINGS_MAX_NAME_LEN + SETTINGS_MAX_VAL_LEN +
+ SETTINGS_EXTRA_LEN];
+ char buf2[SETTINGS_MAX_NAME_LEN + SETTINGS_MAX_VAL_LEN +
+ SETTINGS_EXTRA_LEN];
+ off_t loc1, loc2;
+ char *name1, *val1;
+ char *name2, *val2;
+ int copy;
+ int len, len2;
+ int lines;
+
+ if (fs_open(&rf, cf->cf_name) != 0) {
+ return;
+ }
+
+ settings_tmpfile(tmp_file, cf->cf_name, ".cmp");
+
+ if (settings_file_create_or_replace(&wf, tmp_file)) {
+ fs_close(&rf);
+ return;
+ }
+
+ loc1 = 0;
+ lines = 0;
+ while (1) {
+ len = settings_getnext_line(&rf, buf1, sizeof(buf1), &loc1);
+ if (loc1 == 0 || len < 0) {
+ break;
+ }
+ rc = settings_line_parse(buf1, &name1, &val1);
+ if (rc) {
+ continue;
+ }
+ loc2 = loc1;
+ copy = 1;
+ while ((len2 = settings_getnext_line(&rf, buf2, sizeof(buf2),
+ &loc2)) > 0) {
+ rc = settings_line_parse(buf2, &name2, &val2);
+ if (rc) {
+ continue;
+ }
+ if (!strcmp(name1, name2)) {
+ copy = 0;
+ break;
+ }
+ }
+ if (!copy) {
+ continue;
+ }
+
+ /*
+ * Can't find one. Must copy.
+ */
+ len = settings_line_make(buf2, sizeof(buf2), name1, val1);
+ if (len < 0 || len + 2 > sizeof(buf2)) {
+ continue;
+ }
+ buf2[len++] = '\n';
+ if (fs_write(&wf, buf2, len) != len) {
+ ARG_UNUSED(fs_close(&rf));
+ ARG_UNUSED(fs_close(&wf));
+ return;
+ }
+ lines++;
+ }
+
+ len = fs_close(&wf);
+ len2 = fs_close(&rf);
+ if (len == 0 && len2 == 0 && fs_unlink(cf->cf_name) == 0) {
+ ARG_UNUSED(fs_rename(tmp_file, cf->cf_name));
+ cf->cf_lines = lines;
+ /*
+ * XXX at settings_file_load(), look for .cmp if actual file does not
+ * exist.
+ */
+ }
+}
+
+/*
+ * Called to save configuration.
+ */
+static int settings_file_save(struct settings_store *cs, const char *name,
+ const char *value)
+{
+ struct settings_file *cf = (struct settings_file *)cs;
+ struct fs_file_t file;
+ char buf[SETTINGS_MAX_NAME_LEN + SETTINGS_MAX_VAL_LEN +
+ SETTINGS_EXTRA_LEN];
+ int len;
+ int rc2;
+ int rc;
+
+ if (!name) {
+ return -EINVAL;
+ }
+
+ if (cf->cf_maxlines && (cf->cf_lines + 1 >= cf->cf_maxlines)) {
+ /*
+ * Compress before config file size exceeds
+ * the max number of lines.
+ */
+ settings_file_compress(cf);
+ }
+ len = settings_line_make(buf, sizeof(buf), name, value);
+ if (len < 0 || len + 2 > sizeof(buf)) {
+ return -EINVAL;
+ }
+ buf[len++] = '\n';
+
+ /*
+ * Open the file to add this one value.
+ */
+ rc = fs_open(&file, cf->cf_name);
+ if (rc == 0) {
+ rc = fs_seek(&file, 0, FS_SEEK_END);
+ if (rc == 0) {
+ rc2 = fs_write(&file, buf, len);
+ if (rc2 == len) {
+ cf->cf_lines++;
+ }
+ }
+
+ rc2 = fs_close(&file);
+ if (rc == 0) {
+ rc = rc2;
+ }
+ }
+
+ return rc;
+}
+
+#endif /* #ifdef CONFIG_SETTINGS_FS */
diff --git a/components/bt/ble_mesh/mesh_core/settings/settings_init.c b/components/bt/ble_mesh/mesh_core/settings/settings_init.c
new file mode 100644
index 0000000000..8d2ece4b49
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/settings_init.c
@@ -0,0 +1,88 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+
+#include
+#include
+#include
+
+#include
+#include
+
+#include "sdkconfig.h"
+#include "settings/include/settings.h"
+
+#if CONFIG_BT_MESH_SETTINGS
+
+void settings_init(void);
+
+#ifdef CONFIG_SETTINGS_FS
+#include
+
+static struct settings_file config_init_settings_file = {
+ .cf_name = CONFIG_SETTINGS_FS_FILE,
+ .cf_maxlines = CONFIG_SETTINGS_FS_MAX_LINES
+};
+
+static void settings_init_fs(void)
+{
+ int rc;
+
+ rc = settings_file_src(&config_init_settings_file);
+ if (rc) {
+ BT_ERR("%s, settings file srouce fail.", __func__);
+ return;
+ }
+
+ rc = settings_file_dst(&config_init_settings_file);
+ if (rc) {
+ BT_ERR("%s, settings file dst fail.", __func__);
+ return;
+ }
+}
+
+#elif defined(CONFIG_SETTINGS_FCB)
+#include "settings/include/fcb.h"
+#include "settings/include/settings_fcb.h"
+
+static struct settings_fcb config_init_settings_fcb = {
+
+};
+
+static void settings_init_fcb(void)
+{
+ int rc;
+ rc = settings_fcb_src(&config_init_settings_fcb);
+
+ if (rc != 0) {
+ rc = settings_fcb_src(&config_init_settings_fcb);
+ }
+
+ rc = settings_fcb_dst(&config_init_settings_fcb);
+
+}
+
+#endif
+
+int settings_subsys_init(void)
+{
+ settings_init();
+
+#ifdef CONFIG_SETTINGS_FS
+ settings_init_fs(); /* func rises kernel panic once error */
+
+ /*
+ * Must be called after root FS has been initialized.
+ */
+ return fs_mkdir(CONFIG_SETTINGS_FS_DIR);
+#elif defined(CONFIG_SETTINGS_FCB)
+ settings_init_fcb(); /* func rises kernel panic once error */
+ return 0;
+#endif
+}
+
+#endif /* CONFIG_BT_MESH_SETTINGS */
diff --git a/components/bt/ble_mesh/mesh_core/settings/settings_line.c b/components/bt/ble_mesh/mesh_core/settings/settings_line.c
new file mode 100644
index 0000000000..66d3bc67b7
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/settings_line.c
@@ -0,0 +1,99 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+
+#include "sdkconfig.h"
+
+#include "settings/include/settings.h"
+#include "settings/include/settings_priv.h"
+
+#if CONFIG_BT_MESH_SETTINGS
+
+static inline int isspace(int c)
+{
+ return c == ' ' || ((unsigned)c - '\t') < 5;
+}
+
+int settings_line_parse(char *buf, char **namep, char **valp)
+{
+ char *cp;
+ enum {
+ FIND_NAME,
+ FIND_NAME_END,
+ FIND_VAL,
+ FIND_VAL_END
+ } state = FIND_NAME;
+
+ *valp = NULL;
+ for (cp = buf; *cp != '\0'; cp++) {
+ switch (state) {
+ case FIND_NAME:
+ if (!isspace((unsigned char)*cp)) {
+ *namep = cp;
+ state = FIND_NAME_END;
+ }
+ break;
+ case FIND_NAME_END:
+ if (*cp == '=') {
+ *cp = '\0';
+ state = FIND_VAL;
+ } else if (isspace((unsigned char)*cp)) {
+ *cp = '\0';
+ }
+ break;
+ case FIND_VAL:
+ if (!isspace((unsigned char)*cp)) {
+ *valp = cp;
+ state = FIND_VAL_END;
+ }
+ break;
+ case FIND_VAL_END:
+ if (isspace((unsigned char)*cp)) {
+ *cp = '\0';
+ }
+ break;
+ }
+ }
+
+ if (state == FIND_VAL_END || state == FIND_VAL) {
+ return 0;
+ } else {
+ return -1;
+ }
+}
+
+int settings_line_make(char *dst, int dlen, const char *name, const char *value)
+{
+ int nlen;
+ int vlen;
+ int off;
+
+ nlen = strlen(name);
+
+ if (value) {
+ vlen = strlen(value);
+ } else {
+ vlen = 0;
+ }
+
+ if (nlen + vlen + 2 > dlen) {
+ return -1;
+ }
+
+ memcpy(dst, name, nlen);
+ off = nlen;
+ dst[off++] = '=';
+
+ memcpy(dst + off, value, vlen);
+ off += vlen;
+ dst[off] = '\0';
+
+ return off;
+}
+
+#endif /* CONFIG_BT_MESH_SETTINGS */
diff --git a/components/bt/ble_mesh/mesh_core/settings/settings_store.c b/components/bt/ble_mesh/mesh_core/settings/settings_store.c
new file mode 100644
index 0000000000..1f4c64267b
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/settings/settings_store.c
@@ -0,0 +1,133 @@
+/*
+ * Copyright (c) 2018 Nordic Semiconductor ASA
+ * Copyright (c) 2015 Runtime Inc
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include
+#include
+
+#include "sdkconfig.h"
+
+#include "mesh_util.h"
+#include "mesh_types.h"
+
+#include "settings/include/settings.h"
+#include "settings/include/settings_priv.h"
+#include "settings/include/settings_fcb.h"
+
+#if CONFIG_BT_MESH_SETTINGS
+
+sys_slist_t settings_load_srcs;
+struct settings_store *settings_save_dst;
+
+void settings_src_register(struct settings_store *cs)
+{
+ sys_snode_t *prev, *cur;
+
+ prev = NULL;
+
+ SYS_SLIST_FOR_EACH_NODE(&settings_load_srcs, cur) {
+ prev = cur;
+ }
+
+ sys_slist_insert(&settings_load_srcs, prev, &cs->cs_next);
+}
+
+void settings_dst_register(struct settings_store *cs)
+{
+ settings_save_dst = cs;
+}
+
+static void settings_load_cb(char *name, char *val, void *cb_arg)
+{
+ int rc = settings_set_value(name, val);
+ __ASSERT(rc == 0, "set-value operation failure\n");
+ (void)rc;
+}
+
+int settings_load(void)
+{
+ struct settings_store *cs;
+
+ /*
+ * for every config store
+ * load config
+ * apply config
+ * commit all
+ */
+
+ SYS_SLIST_FOR_EACH_CONTAINER(&settings_load_srcs, cs, cs_next) {
+ cs->cs_itf->csi_load(cs, settings_load_cb, NULL);
+ }
+ return settings_commit(NULL);
+}
+
+/*
+ * Append a single value to persisted config. Don't store duplicate value.
+ */
+int settings_save_one(const char *name, char *value)
+{
+ struct settings_store *cs;
+ struct settings_dup_check_arg cdca;
+
+ cs = settings_save_dst;
+ if (!cs) {
+ return -ENOENT;
+ }
+
+ /*
+ * Check if we're writing the same value again.
+ */
+ cdca.name = name;
+ cdca.val = value;
+ cdca.is_dup = 0;
+ csi_load_check(&cdca);
+ //cs->cs_itf->csi_load(cs, settings_dup_check_cb, &cdca);
+ if (cdca.is_dup == 1) {
+ return 0;
+ }
+ return cs->cs_itf->csi_save(cs, name, value);
+}
+
+int settings_save(void)
+{
+ struct settings_store *cs;
+ struct settings_handler *ch;
+ int rc;
+ int rc2;
+
+ cs = settings_save_dst;
+ if (!cs) {
+ return -ENOENT;
+ }
+
+ if (cs->cs_itf->csi_save_start) {
+ cs->cs_itf->csi_save_start(cs);
+ }
+ rc = 0;
+
+ SYS_SLIST_FOR_EACH_CONTAINER(&settings_handlers, ch, node) {
+ if (ch->h_export) {
+ rc2 = ch->h_export(settings_save_one,
+ SETTINGS_EXPORT_PERSIST);
+ if (!rc) {
+ rc = rc2;
+ }
+ }
+ }
+ if (cs->cs_itf->csi_save_end) {
+ cs->cs_itf->csi_save_end(cs);
+ }
+ return rc;
+}
+
+void settings_store_init(void)
+{
+ sys_slist_init(&settings_load_srcs);
+}
+
+#endif /* CONFIG_BT_MESH_SETTINGS */
diff --git a/components/bt/ble_mesh/mesh_core/test.c b/components/bt/ble_mesh/mesh_core/test.c
new file mode 100644
index 0000000000..1a20427dad
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/test.c
@@ -0,0 +1,131 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+
+#include "mesh_trace.h"
+
+#include "mesh.h"
+#include "test.h"
+#include "crypto.h"
+#include "net.h"
+#include "foundation.h"
+#include "access.h"
+#include "mesh_main.h"
+#include "mesh_access.h"
+
+#if defined(CONFIG_BT_MESH_SELF_TEST)
+
+int bt_mesh_test(void)
+{
+ return 0;
+}
+#endif /* #if defined(CONFIG_BT_MESH_SELF_TEST) */
+
+int bt_mesh_device_auto_enter_network(struct bt_mesh_device_network_info *info)
+{
+ const struct bt_mesh_comp *comp = NULL;
+ struct bt_mesh_model *model = NULL;
+ struct bt_mesh_elem *elem = NULL;
+ struct bt_mesh_app_keys *keys = NULL;
+ struct bt_mesh_app_key *key = NULL;
+ struct bt_mesh_subnet *sub = NULL;
+ int i, j, k;
+ int err;
+
+ if (info == NULL || !BT_MESH_ADDR_IS_UNICAST(info->unicast_addr) ||
+ !BT_MESH_ADDR_IS_GROUP(info->group_addr)) {
+ return -EINVAL;
+ }
+
+ /* The device becomes a node and enters the network */
+ err = bt_mesh_provision(info->net_key, info->net_idx, info->flags, info->iv_index,
+ 0, info->unicast_addr, info->dev_key);
+ if (err) {
+ BT_ERR("%s: bt_mesh_provision() fail, err %d", __func__, err);
+ return err;
+ }
+
+ /* Adds application key to device */
+ sub = bt_mesh_subnet_get(info->net_idx);
+ if (!sub) {
+ BT_ERR("%s: Failed to find subnet 0x%04x", __func__, info->net_idx);
+ return -ENODEV;
+ }
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ key = &bt_mesh.app_keys[i];
+ if (key->net_idx == BT_MESH_KEY_UNUSED) {
+ break;
+ }
+ }
+ if (i == ARRAY_SIZE(bt_mesh.app_keys)) {
+ BT_ERR("%s: Failed to allocate app_key, app_idx 0x%04x", __func__, info->app_idx);
+ return -ENOMEM;
+ }
+
+ keys = sub->kr_flag ? &key->keys[1] : &key->keys[0];
+
+ if (bt_mesh_app_id(info->app_key, &keys->id)) {
+ BT_ERR("%s: Failed to calculate AID, app_idx 0x%04x", __func__, info->app_idx);
+ return -EIO;
+ }
+
+ key->net_idx = info->net_idx;
+ key->app_idx = info->app_idx;
+ memcpy(keys->val, info->app_key, 16);
+
+ /* Binds AppKey with all non-config models, adds group address to all these models */
+ comp = bt_mesh_comp_get();
+ if (!comp) {
+ BT_ERR("%s: Composition data is NULL", __func__);
+ return -ENODEV;
+ }
+
+ for (i = 0; i < comp->elem_count; i++) {
+ elem = &comp->elem[i];
+ for (j = 0; j < elem->model_count; j++) {
+ model = &elem->models[j];
+ if (model->id == BT_MESH_MODEL_ID_CFG_SRV ||
+ model->id == BT_MESH_MODEL_ID_CFG_CLI) {
+ continue;
+ }
+ for (k = 0; k < ARRAY_SIZE(model->keys); k++) {
+ if (model->keys[k] == BT_MESH_KEY_UNUSED) {
+ model->keys[k] = info->app_idx;
+ break;
+ }
+ }
+ for (k = 0; k < ARRAY_SIZE(model->groups); k++) {
+ if (model->groups[k] == BT_MESH_ADDR_UNASSIGNED) {
+ model->groups[k] = info->group_addr;
+ break;
+ }
+ }
+ }
+ for (j = 0; j < elem->vnd_model_count; j++) {
+ model = &elem->vnd_models[j];
+ for (k = 0; k < ARRAY_SIZE(model->keys); k++) {
+ if (model->keys[k] == BT_MESH_KEY_UNUSED) {
+ model->keys[k] = info->app_idx;
+ break;
+ }
+ }
+ for (k = 0; k < ARRAY_SIZE(model->groups); k++) {
+ if (model->groups[k] == BT_MESH_ADDR_UNASSIGNED) {
+ model->groups[k] = info->group_addr;
+ break;
+ }
+ }
+ }
+ }
+
+ return 0;
+}
diff --git a/components/bt/ble_mesh/mesh_core/test.h b/components/bt/ble_mesh/mesh_core/test.h
new file mode 100644
index 0000000000..74a59441bb
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/test.h
@@ -0,0 +1,43 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#ifndef _BLE_MESH_TEST_H
+#define _BLE_MESH_TEST_H
+
+#include
+#include
+#include
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+
+#if defined(CONFIG_BT_MESH_SELF_TEST)
+int bt_mesh_test(void);
+#else
+static inline int bt_mesh_test(void)
+{
+ return 0;
+}
+#endif
+
+struct bt_mesh_device_network_info {
+ u8_t net_key[16];
+ u16_t net_idx;
+ u8_t flags;
+ u32_t iv_index;
+ u16_t unicast_addr;
+ u8_t dev_key[16];
+ u8_t app_key[16];
+ u16_t app_idx;
+ u16_t group_addr;
+};
+
+int bt_mesh_device_auto_enter_network(struct bt_mesh_device_network_info *info);
+
+#endif /* _BLE_MESH_TEST_H */
diff --git a/components/bt/ble_mesh/mesh_core/transport.c b/components/bt/ble_mesh/mesh_core/transport.c
new file mode 100644
index 0000000000..6b81f40628
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/transport.c
@@ -0,0 +1,1618 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+#include "mesh_types.h"
+#include "mesh_util.h"
+
+#include "mesh_buf.h"
+#include "sdkconfig.h"
+#include "settings.h"
+
+#if CONFIG_BT_MESH
+
+#define BT_DBG_ENABLED IS_ENABLED(CONFIG_BT_MESH_DEBUG_TRANS)
+#include "mesh_trace.h"
+#include "mesh_main.h"
+
+#include "crypto.h"
+#include "adv.h"
+#include "mesh.h"
+#include "net.h"
+#include "lpn.h"
+#include "friend.h"
+#include "access.h"
+#include "foundation.h"
+#include "transport.h"
+
+#include "common.h"
+
+#define AID_MASK ((u8_t)(BIT_MASK(6)))
+
+#define SEG(data) ((data)[0] >> 7)
+#define AKF(data) (((data)[0] >> 6) & 0x01)
+#define AID(data) ((data)[0] & AID_MASK)
+#define ASZMIC(data) (((data)[1] >> 7) & 1)
+
+#define APP_MIC_LEN(aszmic) ((aszmic) ? 8 : 4)
+
+#define UNSEG_HDR(akf, aid) ((akf << 6) | (aid & AID_MASK))
+#define SEG_HDR(akf, aid) (UNSEG_HDR(akf, aid) | 0x80)
+
+#define BLOCK_COMPLETE(seg_n) (u32_t)(((u64_t)1 << (seg_n + 1)) - 1)
+
+#define SEQ_AUTH(iv_index, seq) (((u64_t)iv_index) << 24 | (u64_t)seq)
+
+/* Retransmit timeout after which to retransmit unacked segments */
+#define SEG_RETRANSMIT_TIMEOUT K_MSEC(400)
+
+/* Number of retransmit attempts (after the initial transmit) per segment */
+#define SEG_RETRANSMIT_ATTEMPTS 4
+
+/* How long to wait for available buffers before giving up */
+#define BUF_TIMEOUT K_NO_WAIT
+
+static struct seg_tx {
+ struct bt_mesh_subnet *sub;
+ struct net_buf *seg[BT_MESH_TX_SEG_COUNT];
+ u64_t seq_auth;
+ u16_t dst;
+ u8_t seg_n: 5, /* Last segment index */
+ new_key: 1; /* New/old key */
+ u8_t nack_count; /* Number of unacked segs */
+ const struct bt_mesh_send_cb *cb;
+ void *cb_data;
+ struct k_delayed_work retransmit; /* Retransmit timer */
+} seg_tx[CONFIG_BT_MESH_TX_SEG_MSG_COUNT];
+
+static struct seg_rx {
+ struct bt_mesh_subnet *sub;
+ u64_t seq_auth;
+ u8_t seg_n: 5,
+ ctl: 1,
+ in_use: 1,
+ obo: 1;
+ u8_t hdr;
+ u8_t ttl;
+ u16_t src;
+ u16_t dst;
+ u32_t block;
+ u32_t last;
+ struct k_delayed_work ack;
+ struct net_buf_simple buf;
+ u8_t buf_data[CONFIG_BT_MESH_RX_SDU_MAX];
+} seg_rx[CONFIG_BT_MESH_RX_SEG_MSG_COUNT] = {
+ [0 ... (CONFIG_BT_MESH_RX_SEG_MSG_COUNT - 1)] = {
+ .buf.size = CONFIG_BT_MESH_RX_SDU_MAX,
+ },
+};
+
+static u16_t hb_sub_dst = BT_MESH_ADDR_UNASSIGNED;
+
+void bt_mesh_set_hb_sub_dst(u16_t addr)
+{
+ hb_sub_dst = addr;
+}
+
+static int send_unseg(struct bt_mesh_net_tx *tx, struct net_buf_simple *sdu,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ struct net_buf *buf;
+
+ BT_DBG("src 0x%04x dst 0x%04x app_idx 0x%04x sdu_len %u",
+ tx->src, tx->ctx->addr, tx->ctx->app_idx, sdu->len);
+
+ buf = bt_mesh_adv_create(BT_MESH_ADV_DATA,
+ BT_MESH_TRANSMIT_COUNT(tx->xmit),
+ BT_MESH_TRANSMIT_INT(tx->xmit), BUF_TIMEOUT);
+ if (!buf) {
+ BT_ERR("Out of network buffers");
+ return -ENOBUFS;
+ }
+
+ net_buf_reserve(buf, BT_MESH_NET_HDR_LEN);
+
+ if (tx->ctx->app_idx == BT_MESH_KEY_DEV) {
+ net_buf_add_u8(buf, UNSEG_HDR(0, 0));
+ } else {
+ net_buf_add_u8(buf, UNSEG_HDR(1, tx->aid));
+ }
+
+ net_buf_add_mem(buf, sdu->data, sdu->len);
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ if (bt_mesh_friend_enqueue_tx(tx, BT_MESH_FRIEND_PDU_SINGLE,
+ NULL, &buf->b) &&
+ BT_MESH_ADDR_IS_UNICAST(tx->ctx->addr)) {
+ /* PDUs for a specific Friend should only go
+ * out through the Friend Queue.
+ */
+ net_buf_unref(buf);
+ return 0;
+ }
+ }
+ }
+#endif
+
+ return bt_mesh_net_send(tx, buf, cb, cb_data);
+}
+
+bool bt_mesh_tx_in_progress(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(seg_tx); i++) {
+ if (seg_tx[i].nack_count) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+static void seg_tx_reset(struct seg_tx *tx)
+{
+ int i;
+
+ k_delayed_work_cancel(&tx->retransmit);
+
+ tx->cb = NULL;
+ tx->cb_data = NULL;
+ tx->seq_auth = 0;
+ tx->sub = NULL;
+ tx->dst = BT_MESH_ADDR_UNASSIGNED;
+
+ if (!tx->nack_count) {
+ return;
+ }
+
+ for (i = 0; i <= tx->seg_n; i++) {
+ if (!tx->seg[i]) {
+ continue;
+ }
+
+ /** Change by Espressif. Add this to avoid buf->ref is 2 which will
+ * cause lack of buf.
+ */
+ if (tx->seg[i]->ref > 1) {
+ tx->seg[i]->ref = 1;
+ }
+ net_buf_unref(tx->seg[i]);
+ tx->seg[i] = NULL;
+ }
+
+ tx->nack_count = 0;
+
+ if (bt_mesh.pending_update) {
+ BT_DBG("Proceding with pending IV Update");
+ bt_mesh.pending_update = 0;
+ /* bt_mesh_net_iv_update() will re-enable the flag if this
+ * isn't the only transfer.
+ */
+ if (bt_mesh_net_iv_update(bt_mesh.iv_index, false)) {
+ bt_mesh_net_sec_update(NULL);
+ }
+ }
+}
+
+static inline void seg_tx_complete(struct seg_tx *tx, int err)
+{
+ if (tx->cb && tx->cb->end) {
+ tx->cb->end(err, tx->cb_data);
+ }
+
+ seg_tx_reset(tx);
+}
+
+static void seg_send_start(u16_t duration, int err, void *user_data)
+{
+ struct seg_tx *tx = user_data;
+
+ if (tx->cb && tx->cb->start) {
+ tx->cb->start(duration, err, tx->cb_data);
+ }
+}
+
+static void seg_sent(int err, void *user_data)
+{
+ struct seg_tx *tx = user_data;
+
+ k_delayed_work_submit(&tx->retransmit, SEG_RETRANSMIT_TIMEOUT);
+}
+
+static const struct bt_mesh_send_cb first_sent_cb = {
+ .start = seg_send_start,
+ .end = seg_sent,
+};
+
+static const struct bt_mesh_send_cb seg_sent_cb = {
+ .end = seg_sent,
+};
+
+static void seg_tx_send_unacked(struct seg_tx *tx)
+{
+ int i, err;
+
+ for (i = 0; i <= tx->seg_n; i++) {
+ struct net_buf *seg = tx->seg[i];
+
+ if (!seg) {
+ continue;
+ }
+
+ if (BT_MESH_ADV(seg)->busy) {
+ BT_DBG("Skipping segment that's still advertising");
+ continue;
+ }
+
+ if (!(BT_MESH_ADV(seg)->seg.attempts--)) {
+ BT_ERR("Ran out of retransmit attempts");
+ seg_tx_complete(tx, -ETIMEDOUT);
+ return;
+ }
+
+ BT_DBG("resending %u/%u", i, tx->seg_n);
+
+ err = bt_mesh_net_resend(tx->sub, seg, tx->new_key,
+ &seg_sent_cb, tx);
+ if (err) {
+ BT_ERR("Sending segment failed");
+ seg_tx_complete(tx, -EIO);
+ return;
+ }
+ }
+}
+
+static void seg_retransmit(struct k_work *work)
+{
+ struct seg_tx *tx = CONTAINER_OF(work, struct seg_tx, retransmit);
+
+ seg_tx_send_unacked(tx);
+}
+
+static int send_seg(struct bt_mesh_net_tx *net_tx, struct net_buf_simple *sdu,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ u8_t seg_hdr, seg_o;
+ u16_t seq_zero;
+ struct seg_tx *tx;
+ int i;
+
+ BT_DBG("src 0x%04x dst 0x%04x app_idx 0x%04x aszmic %u sdu_len %u",
+ net_tx->src, net_tx->ctx->addr, net_tx->ctx->app_idx,
+ net_tx->aszmic, sdu->len);
+
+ if (sdu->len < 1) {
+ BT_ERR("Zero-length SDU not allowed");
+ return -EINVAL;
+ }
+
+ if (sdu->len > BT_MESH_TX_SDU_MAX) {
+ BT_ERR("Not enough segment buffers for length %u", sdu->len);
+ return -EMSGSIZE;
+ }
+
+ for (tx = NULL, i = 0; i < ARRAY_SIZE(seg_tx); i++) {
+ if (!seg_tx[i].nack_count) {
+ tx = &seg_tx[i];
+ break;
+ }
+ }
+
+ if (!tx) {
+ BT_ERR("No multi-segment message contexts available");
+ return -EBUSY;
+ }
+
+ if (net_tx->ctx->app_idx == BT_MESH_KEY_DEV) {
+ seg_hdr = SEG_HDR(0, 0);
+ } else {
+ seg_hdr = SEG_HDR(1, net_tx->aid);
+ }
+
+ seg_o = 0;
+ tx->dst = net_tx->ctx->addr;
+ tx->seg_n = (sdu->len - 1) / 12;
+ tx->nack_count = tx->seg_n + 1;
+ tx->seq_auth = SEQ_AUTH(BT_MESH_NET_IVI_TX, bt_mesh.seq);
+ tx->sub = net_tx->sub;
+ tx->new_key = net_tx->sub->kr_flag;
+ tx->cb = cb;
+ tx->cb_data = cb_data;
+
+ seq_zero = tx->seq_auth & 0x1fff;
+
+ BT_DBG("SeqZero 0x%04x", seq_zero);
+
+ for (seg_o = 0; sdu->len; seg_o++) {
+ struct net_buf *seg;
+ u16_t len;
+ int err;
+
+ seg = bt_mesh_adv_create(BT_MESH_ADV_DATA,
+ BT_MESH_TRANSMIT_COUNT(net_tx->xmit),
+ BT_MESH_TRANSMIT_INT(net_tx->xmit),
+ BUF_TIMEOUT);
+ if (!seg) {
+ BT_ERR("Out of segment buffers");
+ seg_tx_reset(tx);
+ return -ENOBUFS;
+ }
+
+ BT_MESH_ADV(seg)->seg.attempts = SEG_RETRANSMIT_ATTEMPTS;
+
+ net_buf_reserve(seg, BT_MESH_NET_HDR_LEN);
+
+ net_buf_add_u8(seg, seg_hdr);
+ net_buf_add_u8(seg, (net_tx->aszmic << 7) | seq_zero >> 6);
+ net_buf_add_u8(seg, (((seq_zero & 0x3f) << 2) |
+ (seg_o >> 3)));
+ net_buf_add_u8(seg, ((seg_o & 0x07) << 5) | tx->seg_n);
+
+ len = min(sdu->len, 12);
+ net_buf_add_mem(seg, sdu->data, len);
+ net_buf_simple_pull(sdu, len);
+
+ tx->seg[seg_o] = net_buf_ref(seg);
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ enum bt_mesh_friend_pdu_type type;
+
+ if (seg_o == tx->seg_n) {
+ type = BT_MESH_FRIEND_PDU_COMPLETE;
+ } else {
+ type = BT_MESH_FRIEND_PDU_PARTIAL;
+ }
+
+ if (bt_mesh_friend_enqueue_tx(net_tx, type,
+ &tx->seq_auth,
+ &seg->b) &&
+ BT_MESH_ADDR_IS_UNICAST(net_tx->ctx->addr)) {
+ /* PDUs for a specific Friend should only go
+ * out through the Friend Queue.
+ */
+ net_buf_unref(seg);
+ return 0;
+ }
+ }
+ }
+#endif
+
+ BT_DBG("Sending %u/%u", seg_o, tx->seg_n);
+
+ err = bt_mesh_net_send(net_tx, seg,
+ seg_o ? &seg_sent_cb : &first_sent_cb,
+ tx);
+ if (err) {
+ BT_ERR("Sending segment failed");
+ seg_tx_reset(tx);
+ return err;
+ }
+ }
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (bt_mesh_lpn_established()) {
+ bt_mesh_lpn_poll();
+ }
+ }
+#endif
+
+ return 0;
+}
+
+struct bt_mesh_app_key *bt_mesh_app_key_find(u16_t app_idx)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.app_keys); i++) {
+ struct bt_mesh_app_key *key = &bt_mesh.app_keys[i];
+
+ if (key->net_idx != BT_MESH_KEY_UNUSED &&
+ key->app_idx == app_idx) {
+ return key;
+ }
+ }
+
+ return NULL;
+}
+
+int bt_mesh_trans_send(struct bt_mesh_net_tx *tx, struct net_buf_simple *msg,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ const u8_t *key = NULL;
+ u8_t *ad, role;
+ int err;
+
+ if (net_buf_simple_tailroom(msg) < 4) {
+ BT_ERR("Insufficient tailroom for Transport MIC");
+ return -EINVAL;
+ }
+
+ if (msg->len > 11) {
+ tx->ctx->send_rel = 1;
+ }
+
+ BT_DBG("net_idx 0x%04x app_idx 0x%04x dst 0x%04x", tx->sub->net_idx,
+ tx->ctx->app_idx, tx->ctx->addr);
+ BT_DBG("len %u: %s", msg->len, bt_hex(msg->data, msg->len));
+
+ role = bt_mesh_get_msg_role(tx->ctx->model, tx->ctx->srv_send);
+ if (role == ROLE_NVAL) {
+ BT_ERR("%s: Failed to get role", __func__);
+ return -EINVAL;
+ }
+
+ if (tx->ctx->app_idx == BT_MESH_KEY_DEV) {
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (role == NODE) {
+ if (!bt_mesh_is_provisioner_en()) {
+ key = bt_mesh.dev_key;
+ }
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ key = provisioner_get_device_key(tx->ctx->addr);
+ }
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == NODE) {
+ key = bt_mesh.dev_key;
+ } else if (role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ key = provisioner_get_device_key(tx->ctx->addr);
+ }
+ } else if (role == FAST_PROV) {
+#if CONFIG_BT_MESH_FAST_PROV
+ key = get_fast_prov_device_key(tx->ctx->addr);
+#endif
+ }
+#endif
+
+ if (!key) {
+ BT_ERR("%s: Failed to get device key", __func__);
+ return -EINVAL;
+ }
+
+ tx->aid = 0;
+ } else {
+ struct bt_mesh_app_key *app_key = NULL;
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (role == NODE) {
+ if (!bt_mesh_is_provisioner_en()) {
+ app_key = bt_mesh_app_key_find(tx->ctx->app_idx);
+ }
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ app_key = provisioner_app_key_find(tx->ctx->app_idx);
+ }
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (role == NODE) {
+ app_key = bt_mesh_app_key_find(tx->ctx->app_idx);
+ } else if (role == PROVISIONER) {
+ if (bt_mesh_is_provisioner_en()) {
+ app_key = provisioner_app_key_find(tx->ctx->app_idx);
+ }
+ } else if (role == FAST_PROV) {
+#if CONFIG_BT_MESH_FAST_PROV
+ app_key = get_fast_prov_app_key(tx->ctx->net_idx, tx->ctx->app_idx);
+#endif
+ }
+#endif
+
+ if (!app_key) {
+ BT_ERR("%s: Failed to get app key", __func__);
+ return -EINVAL;
+ }
+
+ if (tx->sub->kr_phase == BT_MESH_KR_PHASE_2 &&
+ app_key->updated) {
+ key = app_key->keys[1].val;
+ tx->aid = app_key->keys[1].id;
+ } else {
+ key = app_key->keys[0].val;
+ tx->aid = app_key->keys[0].id;
+ }
+ }
+
+ if (!tx->ctx->send_rel || net_buf_simple_tailroom(msg) < 8) {
+ tx->aszmic = 0;
+ } else {
+ tx->aszmic = 1;
+ }
+
+ if (BT_MESH_ADDR_IS_VIRTUAL(tx->ctx->addr)) {
+ ad = bt_mesh_label_uuid_get(tx->ctx->addr);
+ } else {
+ ad = NULL;
+ }
+
+ err = bt_mesh_app_encrypt(key, tx->ctx->app_idx == BT_MESH_KEY_DEV,
+ tx->aszmic, msg, ad, tx->src,
+ tx->ctx->addr, bt_mesh.seq,
+ BT_MESH_NET_IVI_TX);
+ if (err) {
+ return err;
+ }
+
+ if (tx->ctx->send_rel) {
+ err = send_seg(tx, msg, cb, cb_data);
+ } else {
+ err = send_unseg(tx, msg, cb, cb_data);
+ }
+
+ return err;
+}
+
+int bt_mesh_trans_resend(struct bt_mesh_net_tx *tx, struct net_buf_simple *msg,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ struct net_buf_simple_state state;
+ int err;
+
+ net_buf_simple_save(msg, &state);
+
+ if (tx->ctx->send_rel || msg->len > 15) {
+ err = send_seg(tx, msg, cb, cb_data);
+ } else {
+ err = send_unseg(tx, msg, cb, cb_data);
+ }
+
+ net_buf_simple_restore(msg, &state);
+
+ return err;
+}
+
+static bool is_replay(struct bt_mesh_net_rx *rx)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(bt_mesh.rpl); i++) {
+ struct bt_mesh_rpl *rpl = &bt_mesh.rpl[i];
+
+ /* Empty slot */
+ if (!rpl->src) {
+ rpl->src = rx->ctx.addr;
+ rpl->seq = rx->seq;
+ rpl->old_iv = rx->old_iv;
+ if (IS_ENABLED(CONFIG_BT_MESH_SETTINGS)) {
+ BT_DBG("Store rpl");
+ bt_mesh_store_rpl(rpl);
+ }
+ return false;
+ }
+
+ /* Existing slot for given address */
+ if (rpl->src == rx->ctx.addr) {
+ if (rx->old_iv && !rpl->old_iv) {
+ return true;
+ }
+
+#if !CONFIG_PATCH_FOR_SILICONLAB_APP_1_1_0
+ if ((!rx->old_iv && rpl->old_iv) ||
+ rpl->seq < rx->seq) {
+#else /* CONFIG_PATCH_FOR_SILICONLAB_APP_1_1_0 */
+ /* Added 10 here to fix the bug of Silicon Lab Android App 1.1.0 when reconnection causess the sequence number to recount from 0. */
+ if ((!rx->old_iv && rpl->old_iv) ||
+ (rpl->seq < rx->seq) || (rpl->seq > rx->seq + 10)) {
+#endif /* #if !CONFIG_PATCH_FOR_SILICONLAB_APP_1_1_0 */
+ rpl->seq = rx->seq;
+ rpl->old_iv = rx->old_iv;
+ if (IS_ENABLED(CONFIG_BT_SETTNGS)) {
+ BT_DBG("Store rpl");
+ bt_mesh_store_rpl(rpl);
+ }
+ return false;
+ } else {
+ return true;
+ }
+ }
+ }
+
+ BT_ERR("RPL is full!");
+ return true;
+}
+
+static int sdu_recv(struct bt_mesh_net_rx *rx, u32_t seq, u8_t hdr, u8_t aszmic,
+ struct net_buf_simple *buf)
+{
+ struct net_buf_simple *sdu =
+ NET_BUF_SIMPLE(CONFIG_BT_MESH_RX_SDU_MAX - 4);
+ u32_t array_size = 0;
+ u8_t *ad;
+ u16_t i;
+ int err;
+
+ BT_DBG("ASZMIC %u AKF %u AID 0x%02x", aszmic, AKF(&hdr), AID(&hdr));
+ BT_DBG("len %u: %s", buf->len, bt_hex(buf->data, buf->len));
+
+ if (buf->len < 1 + APP_MIC_LEN(aszmic)) {
+ BT_ERR("Too short SDU + MIC");
+ return -EINVAL;
+ }
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND) && !rx->local_match) {
+ BT_DBG("Ignoring PDU for LPN 0x%04x of this Friend", rx->dst);
+ return 0;
+ }
+
+ if (BT_MESH_ADDR_IS_VIRTUAL(rx->dst)) {
+ ad = bt_mesh_label_uuid_get(rx->dst);
+ } else {
+ ad = NULL;
+ }
+
+ /* Adjust the length to exclude the MIC at the end */
+ buf->len -= APP_MIC_LEN(aszmic);
+
+ if (!AKF(&hdr)) {
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (!bt_mesh_is_provisioner_en()) {
+ array_size = 1;
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ array_size = 1;
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ array_size = 1;
+ if (bt_mesh_is_provisioner_en()) {
+ array_size += 1;
+ }
+#endif
+
+ for (i = 0; i < array_size; i++) {
+ const u8_t *dev_key = NULL;
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (!bt_mesh_is_provisioner_en()) {
+ dev_key = bt_mesh.dev_key;
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ dev_key = provisioner_get_device_key(rx->ctx.addr);
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (i < 1) {
+ dev_key = bt_mesh.dev_key;
+ } else {
+ dev_key = provisioner_get_device_key(rx->ctx.addr);
+ }
+#endif
+
+ if (!dev_key) {
+ BT_DBG("%s: NULL DevKey", __func__);
+ continue;
+ }
+
+ net_buf_simple_init(sdu, 0);
+ err = bt_mesh_app_decrypt(dev_key, true, aszmic, buf,
+ sdu, ad, rx->ctx.addr, rx->dst,
+ seq, BT_MESH_NET_IVI_RX(rx));
+ if (err) {
+ continue;
+ }
+
+ rx->ctx.app_idx = BT_MESH_KEY_DEV;
+ bt_mesh_model_recv(rx, sdu);
+ return 0;
+ }
+
+ BT_WARN("%s, Unable to decrypt with DevKey", __func__);
+ return -ENODEV;
+ }
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (!bt_mesh_is_provisioner_en()) {
+ array_size = ARRAY_SIZE(bt_mesh.app_keys);
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ array_size = ARRAY_SIZE(bt_mesh.p_app_keys);
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ array_size = ARRAY_SIZE(bt_mesh.app_keys);
+ if (bt_mesh_is_provisioner_en()) {
+ array_size += ARRAY_SIZE(bt_mesh.p_app_keys);
+ }
+#endif
+
+ for (i = 0; i < array_size; i++) {
+ struct bt_mesh_app_key *key = NULL;
+ struct bt_mesh_app_keys *keys = NULL;
+
+#if CONFIG_BT_MESH_NODE && !CONFIG_BT_MESH_PROVISIONER
+ if (!bt_mesh_is_provisioner_en()) {
+ key = &bt_mesh.app_keys[i];
+ }
+#endif
+
+#if !CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (bt_mesh_is_provisioner_en()) {
+ key = bt_mesh.p_app_keys[i];
+ }
+#endif
+
+#if CONFIG_BT_MESH_NODE && CONFIG_BT_MESH_PROVISIONER
+ if (i < ARRAY_SIZE(bt_mesh.app_keys)) {
+ key = &bt_mesh.app_keys[i];
+ } else {
+ key = bt_mesh.p_app_keys[i - ARRAY_SIZE(bt_mesh.app_keys)];
+ }
+#endif
+
+ if (!key) {
+ BT_DBG("%s: NULL AppKey", __func__);
+ continue;
+ }
+
+ /* Make sure that this AppKey matches received net_idx */
+ if (key->net_idx != rx->sub->net_idx) {
+ continue;
+ }
+
+ if (rx->new_key && key->updated) {
+ keys = &key->keys[1];
+ } else {
+ keys = &key->keys[0];
+ }
+
+ /* Make sure that the AppKey ID matches */
+ if (AID(&hdr) != keys->id) {
+ continue;
+ }
+
+ net_buf_simple_init(sdu, 0);
+ err = bt_mesh_app_decrypt(keys->val, false, aszmic, buf,
+ sdu, ad, rx->ctx.addr, rx->dst,
+ seq, BT_MESH_NET_IVI_RX(rx));
+ if (err) {
+ BT_DBG("Unable to decrypt with AppKey 0x%03x", key->app_idx);
+ continue;
+ }
+
+ rx->ctx.app_idx = key->app_idx;
+
+ bt_mesh_model_recv(rx, sdu);
+ return 0;
+ }
+
+ BT_WARN("%s, No matching AppKey", __func__);
+
+ return -EINVAL;
+}
+
+static struct seg_tx *seg_tx_lookup(u16_t seq_zero, u8_t obo, u16_t addr)
+{
+ struct seg_tx *tx;
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(seg_tx); i++) {
+ tx = &seg_tx[i];
+
+ if ((tx->seq_auth & 0x1fff) != seq_zero) {
+ continue;
+ }
+
+ if (tx->dst == addr) {
+ return tx;
+ }
+
+ /* If the expected remote address doesn't match,
+ * but the OBO flag is set and this is the first
+ * acknowledgement, assume it's a Friend that's
+ * responding and therefore accept the message.
+ */
+ if (obo && tx->nack_count == tx->seg_n + 1) {
+ tx->dst = addr;
+ return tx;
+ }
+ }
+
+ return NULL;
+}
+
+static int trans_ack(struct bt_mesh_net_rx *rx, u8_t hdr,
+ struct net_buf_simple *buf, u64_t *seq_auth)
+{
+ struct seg_tx *tx;
+ unsigned int bit;
+ u32_t ack;
+ u16_t seq_zero;
+ u8_t obo;
+
+ if (buf->len < 6) {
+ BT_ERR("Too short ack message");
+ return -EINVAL;
+ }
+
+ seq_zero = net_buf_simple_pull_be16(buf);
+ obo = seq_zero >> 15;
+ seq_zero = (seq_zero >> 2) & 0x1fff;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND) && rx->friend_match) {
+ BT_DBG("Ack for LPN 0x%04x of this Friend", rx->dst);
+ /* Best effort - as there is not enough info for true SeqAuth */
+ *seq_auth = SEQ_AUTH(BT_MESH_NET_IVI_RX(rx), seq_zero);
+ return 0;
+ }
+
+ ack = net_buf_simple_pull_be32(buf);
+
+ BT_DBG("OBO %u seq_zero 0x%04x ack 0x%08x", obo, seq_zero, ack);
+
+ tx = seg_tx_lookup(seq_zero, obo, rx->ctx.addr);
+ if (!tx) {
+ BT_WARN("No matching TX context for ack");
+ return -EINVAL;
+ }
+
+ *seq_auth = tx->seq_auth;
+
+ if (!ack) {
+ BT_WARN("SDU canceled");
+ seg_tx_complete(tx, -ECANCELED);
+ return 0;
+ }
+
+ if (find_msb_set(ack) - 1 > tx->seg_n) {
+ BT_ERR("Too large segment number in ack");
+ return -EINVAL;
+ }
+
+ k_delayed_work_cancel(&tx->retransmit);
+
+ while ((bit = find_lsb_set(ack))) {
+ if (tx->seg[bit - 1]) {
+ BT_DBG("seg %u/%u acked", bit - 1, tx->seg_n);
+ net_buf_unref(tx->seg[bit - 1]);
+ tx->seg[bit - 1] = NULL;
+ tx->nack_count--;
+ }
+
+ ack &= ~BIT(bit - 1);
+ }
+
+ if (tx->nack_count) {
+ seg_tx_send_unacked(tx);
+ } else {
+ BT_DBG("SDU TX complete");
+ seg_tx_complete(tx, 0);
+ }
+
+ return 0;
+}
+
+static int trans_heartbeat(struct bt_mesh_net_rx *rx,
+ struct net_buf_simple *buf)
+{
+ u8_t init_ttl, hops;
+ u16_t feat;
+
+ if (buf->len < 3) {
+ BT_ERR("Too short heartbeat message");
+ return -EINVAL;
+ }
+
+ if (rx->dst != hb_sub_dst) {
+ BT_WARN("Ignoring heartbeat to non-subscribed destination");
+ return 0;
+ }
+
+ init_ttl = (net_buf_simple_pull_u8(buf) & 0x7f);
+ feat = net_buf_simple_pull_be16(buf);
+
+ hops = (init_ttl - rx->ctx.recv_ttl + 1);
+
+ BT_DBG("src 0x%04x TTL %u InitTTL %u (%u hop%s) feat 0x%04x",
+ rx->ctx.addr, rx->ctx.recv_ttl, init_ttl, hops,
+ (hops == 1) ? "" : "s", feat);
+
+ bt_mesh_heartbeat(rx->ctx.addr, rx->dst, hops, feat);
+
+ return 0;
+}
+
+static int ctl_recv(struct bt_mesh_net_rx *rx, u8_t hdr,
+ struct net_buf_simple *buf, u64_t *seq_auth)
+{
+ u8_t ctl_op = TRANS_CTL_OP(&hdr);
+
+ BT_DBG("OpCode 0x%02x len %u", ctl_op, buf->len);
+
+ switch (ctl_op) {
+ case TRANS_CTL_OP_ACK:
+ return trans_ack(rx, hdr, buf, seq_auth);
+ case TRANS_CTL_OP_HEARTBEAT:
+ return trans_heartbeat(rx, buf);
+ }
+
+ /* Only ACKs and heartbeats may need processing without local_match */
+ if (!rx->local_match) {
+ return 0;
+ }
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND) && !bt_mesh_lpn_established()) {
+ switch (ctl_op) {
+ case TRANS_CTL_OP_FRIEND_POLL:
+ return bt_mesh_friend_poll(rx, buf);
+ case TRANS_CTL_OP_FRIEND_REQ:
+ return bt_mesh_friend_req(rx, buf);
+ case TRANS_CTL_OP_FRIEND_CLEAR:
+ return bt_mesh_friend_clear(rx, buf);
+ case TRANS_CTL_OP_FRIEND_CLEAR_CFM:
+ return bt_mesh_friend_clear_cfm(rx, buf);
+ case TRANS_CTL_OP_FRIEND_SUB_ADD:
+ return bt_mesh_friend_sub_add(rx, buf);
+ case TRANS_CTL_OP_FRIEND_SUB_REM:
+ return bt_mesh_friend_sub_rem(rx, buf);
+ }
+ }
+
+#if defined(CONFIG_BT_MESH_LOW_POWER)
+ if (ctl_op == TRANS_CTL_OP_FRIEND_OFFER) {
+ return bt_mesh_lpn_friend_offer(rx, buf);
+ }
+
+ if (rx->ctx.addr == bt_mesh.lpn.frnd) {
+ if (ctl_op == TRANS_CTL_OP_FRIEND_CLEAR_CFM) {
+ return bt_mesh_lpn_friend_clear_cfm(rx, buf);
+ }
+
+ if (!rx->friend_cred) {
+ BT_WARN("Message from a friend with wrong credentials");
+ return -EINVAL;
+ }
+
+ switch (ctl_op) {
+ case TRANS_CTL_OP_FRIEND_UPDATE:
+ return bt_mesh_lpn_friend_update(rx, buf);
+ case TRANS_CTL_OP_FRIEND_SUB_CFM:
+ return bt_mesh_lpn_friend_sub_cfm(rx, buf);
+ }
+ }
+#endif /* CONFIG_BT_MESH_LOW_POWER */
+ }
+#endif /* CONFIG_BT_MESH_NODE */
+
+ BT_WARN("Unhandled TransOpCode 0x%02x", ctl_op);
+
+ return -ENOENT;
+}
+
+static int trans_unseg(struct net_buf_simple *buf, struct bt_mesh_net_rx *rx,
+ u64_t *seq_auth)
+{
+ u8_t hdr;
+
+ BT_DBG("AFK %u AID 0x%02x", AKF(buf->data), AID(buf->data));
+
+ if (buf->len < 1) {
+ BT_ERR("Too small unsegmented PDU");
+ return -EINVAL;
+ }
+
+ if (rx->local_match && is_replay(rx)) {
+ BT_WARN("Replay: src 0x%04x dst 0x%04x seq 0x%06x",
+ rx->ctx.addr, rx->dst, rx->seq);
+ return -EINVAL;
+ }
+
+ hdr = net_buf_simple_pull_u8(buf);
+
+ if (rx->ctl) {
+ return ctl_recv(rx, hdr, buf, seq_auth);
+ } else {
+ /* SDUs must match a local element or an LPN of this Friend. */
+ if (!rx->local_match && !rx->friend_match) {
+ return 0;
+ }
+
+ return sdu_recv(rx, rx->seq, hdr, 0, buf);
+ }
+}
+
+static inline s32_t ack_timeout(struct seg_rx *rx)
+{
+ s32_t to;
+
+ /* The acknowledgment timer shall be set to a minimum of
+ * 150 + 50 * TTL milliseconds.
+ */
+ to = K_MSEC(150 + (50 * rx->ttl));
+
+ /* 100 ms for every not yet received segment */
+ to += K_MSEC(((rx->seg_n + 1) - popcount(rx->block)) * 100);
+
+ /* Make sure we don't send more frequently than the duration for
+ * each packet (default is 300ms).
+ */
+ return max(to, K_MSEC(400));
+}
+
+int bt_mesh_ctl_send(struct bt_mesh_net_tx *tx, u8_t ctl_op, void *data,
+ size_t data_len, u64_t *seq_auth,
+ const struct bt_mesh_send_cb *cb, void *cb_data)
+{
+ struct net_buf *buf;
+
+ BT_DBG("src 0x%04x dst 0x%04x ttl 0x%02x ctl 0x%02x", tx->src,
+ tx->ctx->addr, tx->ctx->send_ttl, ctl_op);
+ BT_DBG("len %u: %s", data_len, bt_hex(data, data_len));
+
+ buf = bt_mesh_adv_create(BT_MESH_ADV_DATA,
+ BT_MESH_TRANSMIT_COUNT(tx->xmit),
+ BT_MESH_TRANSMIT_INT(tx->xmit), BUF_TIMEOUT);
+ if (!buf) {
+ BT_ERR("Out of transport buffers");
+ return -ENOBUFS;
+ }
+
+ net_buf_reserve(buf, BT_MESH_NET_HDR_LEN);
+
+ net_buf_add_u8(buf, TRANS_CTL_HDR(ctl_op, 0));
+
+ net_buf_add_mem(buf, data, data_len);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ if (bt_mesh_friend_enqueue_tx(tx, BT_MESH_FRIEND_PDU_SINGLE,
+ seq_auth, &buf->b) &&
+ BT_MESH_ADDR_IS_UNICAST(tx->ctx->addr)) {
+ /* PDUs for a specific Friend should only go
+ * out through the Friend Queue.
+ */
+ net_buf_unref(buf);
+ return 0;
+ }
+ }
+
+ return bt_mesh_net_send(tx, buf, cb, cb_data);
+}
+
+static int send_ack(struct bt_mesh_subnet *sub, u16_t src, u16_t dst,
+ u8_t ttl, u64_t *seq_auth, u32_t block, u8_t obo)
+{
+ struct bt_mesh_msg_ctx ctx = {
+ .net_idx = sub->net_idx,
+ .app_idx = BT_MESH_KEY_UNUSED,
+ .addr = dst,
+ .send_ttl = ttl,
+ };
+ struct bt_mesh_net_tx tx = {
+ .sub = sub,
+ .ctx = &ctx,
+ .src = obo ? bt_mesh_primary_addr() : src,
+ .xmit = bt_mesh_net_transmit_get(),
+ };
+ u16_t seq_zero = *seq_auth & 0x1fff;
+ u8_t buf[6];
+
+ BT_DBG("SeqZero 0x%04x Block 0x%08x OBO %u", seq_zero, block, obo);
+
+ if (bt_mesh_lpn_established()) {
+ BT_WARN("Not sending ACK when LPN is enabled");
+ return 0;
+ }
+
+ /* This can happen if the segmented message was destined for a group
+ * or a virtual address.
+ */
+ if (!BT_MESH_ADDR_IS_UNICAST(src)) {
+ BT_WARN("Not sending ACK for non-unicast address");
+ return 0;
+ }
+
+ sys_put_be16(((seq_zero << 2) & 0x7ffc) | (obo << 15), buf);
+ sys_put_be32(block, &buf[2]);
+
+ return bt_mesh_ctl_send(&tx, TRANS_CTL_OP_ACK, buf, sizeof(buf),
+ NULL, NULL, NULL);
+}
+
+static void seg_rx_reset(struct seg_rx *rx)
+{
+ BT_DBG("rx %p", rx);
+
+ k_delayed_work_cancel(&rx->ack);
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND) && rx->obo &&
+ rx->block != BLOCK_COMPLETE(rx->seg_n)) {
+ BT_WARN("Clearing incomplete buffers from Friend queue");
+ bt_mesh_friend_clear_incomplete(rx->sub, rx->src, rx->dst,
+ &rx->seq_auth);
+ }
+
+ /* rx->net and rx->seq_auth are not reset here since we need to
+ * be able to send an ACK if we receive a segment after the full SDU
+ * has already been received.
+ */
+
+ rx->in_use = 0;
+}
+
+static void seg_ack(struct k_work *work)
+{
+ struct seg_rx *rx = CONTAINER_OF(work, struct seg_rx, ack);
+
+ BT_DBG("rx %p", rx);
+
+ if (k_uptime_get_32() - rx->last > (60 * MSEC_PER_SEC)) {
+ BT_WARN("Incomplete timer expired");
+ send_ack(rx->sub, rx->dst, rx->src, rx->ttl,
+ &rx->seq_auth, 0, rx->obo);
+ seg_rx_reset(rx);
+ return;
+ }
+
+ send_ack(rx->sub, rx->dst, rx->src, rx->ttl, &rx->seq_auth,
+ rx->block, rx->obo);
+
+ k_delayed_work_submit(&rx->ack, ack_timeout(rx));
+}
+
+static inline u8_t seg_len(bool ctl)
+{
+ if (ctl) {
+ return 8;
+ } else {
+ return 12;
+ }
+}
+
+static inline bool sdu_len_is_ok(bool ctl, u8_t seg_n)
+{
+ return ((seg_n * seg_len(ctl) + 1) <= CONFIG_BT_MESH_RX_SDU_MAX);
+}
+
+static struct seg_rx *seg_rx_find(struct bt_mesh_net_rx *net_rx,
+ const u64_t *seq_auth)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(seg_rx); i++) {
+ struct seg_rx *rx = &seg_rx[i];
+
+ if (rx->src != net_rx->ctx.addr || rx->dst != net_rx->dst) {
+ continue;
+ }
+
+ if (rx->seq_auth == *seq_auth) {
+ return rx;
+ }
+
+ if (rx->in_use) {
+ BT_WARN("Duplicate SDU from src 0x%04x",
+ net_rx->ctx.addr);
+
+ /* Clear out the old context since the sender
+ * has apparently started sending a new SDU.
+ */
+ seg_rx_reset(rx);
+
+ /* Return non-match so that the caller can re-allocate */
+ return NULL;
+ }
+ }
+
+ return NULL;
+}
+
+static bool seg_rx_is_valid(struct seg_rx *rx, struct bt_mesh_net_rx *net_rx,
+ const u8_t *hdr, u8_t seg_n)
+{
+ if (rx->hdr != *hdr || rx->seg_n != seg_n) {
+ BT_ERR("Invalid segment for ongoing session");
+ return false;
+ }
+
+ if (rx->src != net_rx->ctx.addr || rx->dst != net_rx->dst) {
+ BT_ERR("Invalid source or destination for segment");
+ return false;
+ }
+
+ if (rx->ctl != net_rx->ctl) {
+ BT_ERR("Inconsistent CTL in segment");
+ return false;
+ }
+
+ return true;
+}
+
+static struct seg_rx *seg_rx_alloc(struct bt_mesh_net_rx *net_rx,
+ const u8_t *hdr, const u64_t *seq_auth,
+ u8_t seg_n)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(seg_rx); i++) {
+ struct seg_rx *rx = &seg_rx[i];
+
+ if (rx->in_use) {
+ continue;
+ }
+
+ rx->in_use = 1;
+ net_buf_simple_init(&rx->buf, 0);
+ rx->sub = net_rx->sub;
+ rx->ctl = net_rx->ctl;
+ rx->seq_auth = *seq_auth;
+ rx->seg_n = seg_n;
+ rx->hdr = *hdr;
+ rx->ttl = net_rx->ctx.send_ttl;
+ rx->src = net_rx->ctx.addr;
+ rx->dst = net_rx->dst;
+ rx->block = 0;
+
+ BT_DBG("New RX context. Block Complete 0x%08x",
+ BLOCK_COMPLETE(seg_n));
+
+ return rx;
+ }
+
+ return NULL;
+}
+
+static int trans_seg(struct net_buf_simple *buf, struct bt_mesh_net_rx *net_rx,
+ enum bt_mesh_friend_pdu_type *pdu_type, u64_t *seq_auth)
+{
+ struct seg_rx *rx;
+ u8_t *hdr = buf->data;
+ u16_t seq_zero;
+ u8_t seg_n;
+ u8_t seg_o;
+ int err;
+
+ if (buf->len < 5) {
+ BT_ERR("Too short segmented message (len %u)", buf->len);
+ return -EINVAL;
+ }
+
+ BT_DBG("ASZMIC %u AKF %u AID 0x%02x", ASZMIC(hdr), AKF(hdr), AID(hdr));
+
+ net_buf_simple_pull(buf, 1);
+
+ seq_zero = net_buf_simple_pull_be16(buf);
+ seg_o = (seq_zero & 0x03) << 3;
+ seq_zero = (seq_zero >> 2) & 0x1fff;
+ seg_n = net_buf_simple_pull_u8(buf);
+ seg_o |= seg_n >> 5;
+ seg_n &= 0x1f;
+
+ BT_DBG("SeqZero 0x%04x SegO %u SegN %u", seq_zero, seg_o, seg_n);
+
+ if (seg_o > seg_n) {
+ BT_ERR("SegO greater than SegN (%u > %u)", seg_o, seg_n);
+ return -EINVAL;
+ }
+
+ /* According to Mesh 1.0 specification:
+ * "The SeqAuth is composed of the IV Index and the sequence number
+ * (SEQ) of the first segment"
+ *
+ * Therefore we need to calculate very first SEQ in order to find
+ * seqAuth. We can calculate as below:
+ *
+ * SEQ(0) = SEQ(n) - (delta between seqZero and SEQ(n) by looking into
+ * 14 least significant bits of SEQ(n))
+ *
+ * Mentioned delta shall be >= 0, if it is not then seq_auth will
+ * be broken and it will be verified by the code below.
+ */
+ *seq_auth = SEQ_AUTH(BT_MESH_NET_IVI_RX(net_rx),
+ (net_rx->seq -
+ ((((net_rx->seq & BIT_MASK(14)) - seq_zero)) &
+ BIT_MASK(13))));
+
+ /* Look for old RX sessions */
+ rx = seg_rx_find(net_rx, seq_auth);
+ if (rx) {
+ /* Discard old SeqAuth packet */
+ if (rx->seq_auth > *seq_auth) {
+ BT_WARN("Ignoring old SeqAuth");
+ return -EINVAL;
+ }
+
+ if (!seg_rx_is_valid(rx, net_rx, hdr, seg_n)) {
+ return -EINVAL;
+ }
+
+ if (rx->in_use) {
+ BT_DBG("Existing RX context. Block 0x%08x", rx->block);
+ goto found_rx;
+ }
+
+ if (rx->block == BLOCK_COMPLETE(rx->seg_n)) {
+ BT_WARN("Got segment for already complete SDU");
+ send_ack(net_rx->sub, net_rx->dst, net_rx->ctx.addr,
+ net_rx->ctx.send_ttl, seq_auth, rx->block,
+ rx->obo);
+ return -EALREADY;
+ }
+
+ /* We ignore instead of sending block ACK 0, since the
+ * ACK timer is always smaller than the incomplete
+ * timer, i.e. the sender is misbehaving.
+ */
+ BT_WARN("Got segment for canceled SDU");
+ return -EINVAL;
+ }
+
+ /* Leave early, if such a large SDU cannot be received */
+ if (!sdu_len_is_ok(net_rx->ctl, seg_n)) {
+ BT_ERR("Too big incoming SDU length");
+ send_ack(net_rx->sub, net_rx->dst, net_rx->ctx.addr,
+ net_rx->ctx.send_ttl, seq_auth, 0,
+ net_rx->friend_match);
+ return -EMSGSIZE;
+ }
+
+ /* Look for a free slot for a new RX session */
+ rx = seg_rx_alloc(net_rx, hdr, seq_auth, seg_n);
+ if (!rx) {
+ /* Give a warning but don't cancel as the existing slots
+ * will eventually be freed up to make processing of
+ * this one possible.
+ */
+ BT_WARN("No free slots for new incoming segmented messages");
+ return -ENOMEM;
+ }
+
+ rx->obo = net_rx->friend_match;
+
+found_rx:
+ if (BIT(seg_o) & rx->block) {
+ BT_WARN("Received already received fragment");
+ return -EALREADY;
+ }
+
+ /* All segments, except for the last one, must either have 8 bytes
+ * of payload (for 64bit Net MIC) or 12 bytes of payload (for 32bit
+ * Net MIC).
+ */
+ if (seg_o == seg_n) {
+ /* Set the expected final buffer length */
+ rx->buf.len = seg_n * seg_len(rx->ctl) + buf->len;
+ BT_DBG("Target len %u * %u + %u = %u", seg_n, seg_len(rx->ctl),
+ buf->len, rx->buf.len);
+
+ if (rx->buf.len > CONFIG_BT_MESH_RX_SDU_MAX) {
+ BT_ERR("Too large SDU len");
+ send_ack(net_rx->sub, net_rx->dst, net_rx->ctx.addr,
+ net_rx->ctx.send_ttl, seq_auth, 0, rx->obo);
+ seg_rx_reset(rx);
+ return -EMSGSIZE;
+ }
+ } else {
+ if (buf->len != seg_len(rx->ctl)) {
+ BT_ERR("Incorrect segment size for message type");
+ return -EINVAL;
+ }
+ }
+
+ /* Reset the Incomplete Timer */
+ rx->last = k_uptime_get_32();
+
+ if (!k_delayed_work_remaining_get(&rx->ack) &&
+ !bt_mesh_lpn_established()) {
+ k_delayed_work_submit(&rx->ack, ack_timeout(rx));
+ }
+
+ /* Location in buffer can be calculated based on seg_o & rx->ctl */
+ memcpy(rx->buf_data + (seg_o * seg_len(rx->ctl)), buf->data, buf->len);
+
+ BT_DBG("Received %u/%u", seg_o, seg_n);
+
+ /* Mark segment as received */
+ rx->block |= BIT(seg_o);
+
+ if (rx->block != BLOCK_COMPLETE(seg_n)) {
+ *pdu_type = BT_MESH_FRIEND_PDU_PARTIAL;
+ return 0;
+ }
+
+ BT_DBG("Complete SDU");
+
+ if (net_rx->local_match && is_replay(net_rx)) {
+ BT_WARN("Replay: src 0x%04x dst 0x%04x seq 0x%06x",
+ net_rx->ctx.addr, net_rx->dst, net_rx->seq);
+ /* Clear the segment's bit */
+ rx->block &= ~BIT(seg_o);
+ return -EINVAL;
+ }
+
+ *pdu_type = BT_MESH_FRIEND_PDU_COMPLETE;
+
+ k_delayed_work_cancel(&rx->ack);
+ send_ack(net_rx->sub, net_rx->dst, net_rx->ctx.addr,
+ net_rx->ctx.send_ttl, seq_auth, rx->block, rx->obo);
+
+ if (net_rx->ctl) {
+ err = ctl_recv(net_rx, *hdr, &rx->buf, seq_auth);
+ } else {
+ err = sdu_recv(net_rx, (rx->seq_auth & 0xffffff), *hdr, ASZMIC(hdr), &rx->buf);
+ }
+
+ seg_rx_reset(rx);
+
+ return err;
+}
+
+int bt_mesh_trans_recv(struct net_buf_simple *buf, struct bt_mesh_net_rx *rx)
+{
+ u64_t seq_auth = TRANS_SEQ_AUTH_NVAL;
+ enum bt_mesh_friend_pdu_type pdu_type = BT_MESH_FRIEND_PDU_SINGLE;
+ struct net_buf_simple_state state;
+ int err;
+
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND)) {
+ rx->friend_match = bt_mesh_friend_match(rx->sub->net_idx,
+ rx->dst);
+ } else {
+ rx->friend_match = false;
+ }
+
+ BT_DBG("src 0x%04x dst 0x%04x seq 0x%08x friend_match %u",
+ rx->ctx.addr, rx->dst, rx->seq, rx->friend_match);
+
+ /* Remove network headers */
+ net_buf_simple_pull(buf, BT_MESH_NET_HDR_LEN);
+
+ BT_DBG("Payload %s", bt_hex(buf->data, buf->len));
+
+ /* If LPN mode is enabled, messages are only accepted when
+ * the Friend is requested to send them. The messages must
+ * also be encrypted using the Friend Credentials.
+ */
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER) &&
+ bt_mesh_lpn_established() && rx->net_if == BT_MESH_NET_IF_ADV &&
+ (!bt_mesh_lpn_waiting_update() || !rx->friend_cred)) {
+ BT_WARN("Ignoring unexpected message in Low Power mode");
+ return -EAGAIN;
+ }
+ }
+#endif
+
+ /* Save the app-level state so the buffer can later be placed in
+ * the Friend Queue.
+ */
+ net_buf_simple_save(buf, &state);
+
+ if (SEG(buf->data)) {
+ /* Segmented messages must match a local element or an
+ * LPN of this Friend.
+ */
+ if (!rx->local_match && !rx->friend_match) {
+ return 0;
+ }
+
+ err = trans_seg(buf, rx, &pdu_type, &seq_auth);
+ } else {
+ err = trans_unseg(buf, rx, &seq_auth);
+ }
+
+ /* Notify LPN state machine, so that a Friend Poll can be sent. If the
+ * message was a Friend Update it's possible that a Poll was already
+ * queued for sending. However that's fine since the
+ * bt_mesh_lpn_waiting_update() function will then return false:
+ * we still need to go through the actual sending to the bearer and
+ * wait for ReceiveDelay before transitioning to WAIT_UPDATE state.
+ *
+ * Another situation where we want to notify the LPN state machine
+ * is when it's configured to use an automatic Friendship establishment
+ * timer. In this case, the timer must be reset at this point.
+ *
+ * ENOENT is a special condition which is only used to indicate that
+ * the Transport OpCode was invalid. In this case, the PDU should be
+ * ignored completely, as per MESH/NODE/FRND/LPN/BI-02-C.
+ */
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_LOW_POWER) && err != -ENOENT &&
+ (bt_mesh_lpn_timer() ||
+ (bt_mesh_lpn_established() && bt_mesh_lpn_waiting_update()))) {
+ bt_mesh_lpn_msg_received(rx);
+ }
+ }
+#endif
+
+ net_buf_simple_restore(buf, &state);
+
+#if CONFIG_BT_MESH_NODE
+ if (bt_mesh_is_provisioned()) {
+ if (IS_ENABLED(CONFIG_BT_MESH_FRIEND) && rx->friend_match && !err) {
+ if (seq_auth == TRANS_SEQ_AUTH_NVAL) {
+ bt_mesh_friend_enqueue_rx(rx, pdu_type, NULL, buf);
+ } else {
+ bt_mesh_friend_enqueue_rx(rx, pdu_type, &seq_auth, buf);
+ }
+ }
+ }
+#endif
+
+ return err;
+}
+
+void bt_mesh_rx_reset(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(seg_rx); i++) {
+ seg_rx_reset(&seg_rx[i]);
+ seg_rx[i].src = BT_MESH_ADDR_UNASSIGNED;
+ seg_rx[i].dst = BT_MESH_ADDR_UNASSIGNED;
+ }
+}
+
+void bt_mesh_tx_reset(void)
+{
+ int i;
+
+ BT_DBG("%s", __func__);
+
+ for (i = 0; i < ARRAY_SIZE(seg_tx); i++) {
+ seg_tx_reset(&seg_tx[i]);
+ }
+}
+
+void bt_mesh_trans_init(void)
+{
+ int i;
+
+ for (i = 0; i < ARRAY_SIZE(seg_tx); i++) {
+ k_delayed_work_init(&seg_tx[i].retransmit, seg_retransmit);
+ }
+
+ for (i = 0; i < ARRAY_SIZE(seg_rx); i++) {
+ k_delayed_work_init(&seg_rx[i].ack, seg_ack);
+ }
+}
+
+void bt_mesh_rpl_clear(void)
+{
+ BT_DBG("%s", __func__);
+ memset(bt_mesh.rpl, 0, sizeof(bt_mesh.rpl));
+}
+
+#endif /* #if CONFIG_BT_MESH */
diff --git a/components/bt/ble_mesh/mesh_core/transport.h b/components/bt/ble_mesh/mesh_core/transport.h
new file mode 100644
index 0000000000..f44ade772d
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_core/transport.h
@@ -0,0 +1,98 @@
+/* Bluetooth Mesh */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#define TRANS_SEQ_AUTH_NVAL 0xffffffffffffffff
+
+#define BT_MESH_TX_SEG_COUNT (CONFIG_BT_MESH_ADV_BUF_COUNT - 3)
+#define BT_MESH_TX_SDU_MAX (BT_MESH_TX_SEG_COUNT * 12)
+
+#define TRANS_CTL_OP_MASK ((u8_t)BIT_MASK(7))
+#define TRANS_CTL_OP(data) ((data)[0] & TRANS_CTL_OP_MASK)
+#define TRANS_CTL_HDR(op, seg) ((op & TRANS_CTL_OP_MASK) | (seg << 7))
+
+#define TRANS_CTL_OP_ACK 0x00
+#define TRANS_CTL_OP_FRIEND_POLL 0x01
+#define TRANS_CTL_OP_FRIEND_UPDATE 0x02
+#define TRANS_CTL_OP_FRIEND_REQ 0x03
+#define TRANS_CTL_OP_FRIEND_OFFER 0x04
+#define TRANS_CTL_OP_FRIEND_CLEAR 0x05
+#define TRANS_CTL_OP_FRIEND_CLEAR_CFM 0x06
+#define TRANS_CTL_OP_FRIEND_SUB_ADD 0x07
+#define TRANS_CTL_OP_FRIEND_SUB_REM 0x08
+#define TRANS_CTL_OP_FRIEND_SUB_CFM 0x09
+#define TRANS_CTL_OP_HEARTBEAT 0x0a
+
+struct bt_mesh_ctl_friend_poll {
+ u8_t fsn;
+} __packed;
+
+struct bt_mesh_ctl_friend_update {
+ u8_t flags;
+ u32_t iv_index;
+ u8_t md;
+} __packed;
+
+struct bt_mesh_ctl_friend_req {
+ u8_t criteria;
+ u8_t recv_delay;
+ u8_t poll_to[3];
+ u16_t prev_addr;
+ u8_t num_elem;
+ u16_t lpn_counter;
+} __packed;
+
+struct bt_mesh_ctl_friend_offer {
+ u8_t recv_win;
+ u8_t queue_size;
+ u8_t sub_list_size;
+ s8_t rssi;
+ u16_t frnd_counter;
+} __packed;
+
+struct bt_mesh_ctl_friend_clear {
+ u16_t lpn_addr;
+ u16_t lpn_counter;
+} __packed;
+
+struct bt_mesh_ctl_friend_clear_confirm {
+ u16_t lpn_addr;
+ u16_t lpn_counter;
+} __packed;
+
+#define BT_MESH_FRIEND_SUB_MIN_LEN (1 + 2)
+struct bt_mesh_ctl_friend_sub {
+ u8_t xact;
+ u16_t addr_list[5];
+} __packed;
+
+struct bt_mesh_ctl_friend_sub_confirm {
+ u8_t xact;
+} __packed;
+
+void bt_mesh_set_hb_sub_dst(u16_t addr);
+
+struct bt_mesh_app_key *bt_mesh_app_key_find(u16_t app_idx);
+
+bool bt_mesh_tx_in_progress(void);
+
+void bt_mesh_rx_reset(void);
+void bt_mesh_tx_reset(void);
+
+int bt_mesh_ctl_send(struct bt_mesh_net_tx *tx, u8_t ctl_op, void *data,
+ size_t data_len, u64_t *seq_auth,
+ const struct bt_mesh_send_cb *cb, void *cb_data);
+
+int bt_mesh_trans_send(struct bt_mesh_net_tx *tx, struct net_buf_simple *msg,
+ const struct bt_mesh_send_cb *cb, void *cb_data);
+
+int bt_mesh_trans_recv(struct net_buf_simple *buf, struct bt_mesh_net_rx *rx);
+
+void bt_mesh_trans_init(void);
+
+void bt_mesh_rpl_clear(void);
diff --git a/components/bt/ble_mesh/mesh_docs/BLE-Mesh_FAQs_EN.md b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_FAQs_EN.md
new file mode 100644
index 0000000000..3e3ba69684
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_FAQs_EN.md
@@ -0,0 +1,9 @@
+# Frequently Asked Questions
+
+## General Questions
+
+### Why I do not get a reply from the remote device when I perform get operation immediately after set operation has been performed?
+* Any Client Model operation needs to wait for the completion event of an ongoing operation. Once the completion event is received the next command can be executed. If a command is executed before the completion event is received, a timeout error will occur.
+
+### When I use the API `esp_ble_mesh_client_model_send_msg`, why does it crash with the log messages *Invalid client value when sent client msg* or *Invalid client value when sent client msg*?
+* You should initialize a structure of the type `esp_ble_mesh_client_t` and set its value as the user data of client model.
diff --git a/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Feature_List_EN.md b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Feature_List_EN.md
new file mode 100644
index 0000000000..cf54f7985e
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Feature_List_EN.md
@@ -0,0 +1,89 @@
+# Espressif BLE Mesh Feature List
+
+## Currently Supported Features
+
+### Mesh Core
+
+* Provisioning: Node Role
+ * Advertising and GATT bearer
+ * Authentication OOB
+
+* Provisioning: Provisioner Role
+ * Advertising and GATT bearer
+ * Authentication OOB
+
+* Networking
+ * Relay
+ * Segmentation and Reassembly
+ * Key Refresh
+ * IV Update
+
+* Proxy Support
+
+* Multiple Client Models Run Simultaneously
+ * Support multiple client models send packets to different nodes simultaneously
+ * No blocking between client model and server
+
+* NVS Storing
+ * Store Provisioning Data of The Node Device
+
+### Mesh Applications
+
+* Fast Provisioning
+ * Fast Provisioning Server Model
+ * Fast Provisioning Client Model
+ * Example & Demo Video
+
+* Wi-Fi & BLE Mesh Coexistence
+ * Example & Demo Video(coming soon)
+
+* Mesh Console Commands
+ * Example
+
+
+### Mesh Models
+
+* Foundation Models
+ * Configuration Server Model
+ * Configuration Client Model
+ * Health Server Model
+ * Health Client Model
+
+* Generic Client Models
+ * Generic OnOff Client
+ * Generic Level Client
+ * Generic Location Client
+ * Generic Default Transition Timer Client
+ * Generic Power OnOff Client
+ * Generic Power Level Client
+ * Generic Battery Client
+ * Generic Property Client
+
+* Generic Server Models
+ * Generic OnOff Server (Simple)
+
+* Lighting Client Models
+ * Light Lightness Client
+ * Light CTL Client
+ * Light HSL Client
+
+* Sensor Client Models
+ * Sensor Client
+
+* Time and Scenes Client Models
+ * Scene Client
+
+
+## Future Release Features
+
+### Mesh Core
+
+* BLE Mesh BQB Certification
+* Friend Feature
+* Low Power Node Feature
+
+### Mesh Applications
+
+* Fast OTA
+
+### Mesh Models
diff --git a/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Getting_Started_EN.md b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Getting_Started_EN.md
new file mode 100644
index 0000000000..7e6ca5a39c
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Getting_Started_EN.md
@@ -0,0 +1,143 @@
+# Introduction
+
+Bluetooth mesh networking enables many-to-many (m:m) device communications and is optimized for creating large-scale device networks.
+
+Devices may relay data to other devices not in direct radio range of the originating device. In this way, mesh networks can span very large physical areas and contain large numbers of devices. It is ideally suited for building automation, sensor networks, and other IoT solutions where tens, hundreds, or thousands of devices need to reliably and securely communicate with one another.
+
+Bluetooth mesh is not a wireless communications technology, but a networking technology. This technology is dependent upon Bluetooth Low Energy (BLE) - a wireless communications protocol stack.
+
+
+# Specifications
+
+The official specifications for Bluetooth mesh can be found [here](https://www.bluetooth.com/specifications/mesh-specifications)
+
+
+# Getting Started with BLE Mesh on ESP32
+
+If you are new to ESP32, you may first need to go through the [Getting Started guide](https://docs.espressif.com/projects/esp-idf/en/latest/get-started/index.html).
+
+Built on top of Zephyr BLE Mesh stack, the ESP BLE Mesh implementation supports device provisioning and node control. It also supports such node features as Proxy, Relay, Low power and Friend.
+
+
+## Access to ESP BLE Mesh
+Since you are on this page, you should already have access to Espressif BLE Mesh SDK. If you do not have access, please get in touch with your point of contact.
+
+## Documentation
+
+The ESP BLE Mesh code in the SDK is organized as below. Each folder contains source files related to it as well as a subfolder with header files for the exposed functionality.
+
+```
+$tree components/bt/ble_mesh/
+
+├── api /* BLE Mesh functionality exposed through esp_ble_mesh_* APIs for the applications */
+│ ├── core /* BLE Mesh Core APIs */
+│ │ └── include
+│ └── models /* Foundation Models and other Client Models APIs */
+│ └── include
+├── btc
+│ └── include
+├── mesh_core /* BLE mesh core based on Zephyr BLE stack with miscellaneous modifications and
+│ │ an adaptation layer to make it work with ESP32 */
+│ └── include
+├── mesh_docs /* BLE Mesh docs */
+└── mesh_models /* Foundation Models and other Client Models implementations */
+ └── include
+```
+
+To demonstrate the features supported by BLE Mesh SDK, a few sample examples have been added. Each example has a README.md file for quick start as well as a walkthrough file that explains the functionality in detail.
+
+Below is a snapshot of the BLE Mesh examples directory
+
+```
+$ tree examples/bluetooth/ble_mesh/
+├── ble_mesh_client_model
+│ ├── main
+│ │ ├── ble_mesh_client_model_main.c
+│ │ ├── board.c
+│ │ ├── board.h
+│ │ ├── component.mk
+│ │ └── Kconfig.projbuild
+│ ├── Makefile
+│ ├── README.md
+│ └── sdkconfig.defaults
+├── ble_mesh_node
+│ ├── main
+│ │ ├── ble_mesh_demo_main.c
+│ │ ├── board.c
+│ │ ├── board.h
+│ │ ├── component.mk
+│ │ └── Kconfig.projbuild
+│ ├── Makefile
+│ ├── README.md
+│ ├── sdkconfig.defaults
+│ └── tutorial
+│ └── Ble_Mesh_Node_Example_Walkthrough.md
+└── ble_mesh_provisioner
+ ├── main
+ │ ├── ble_mesh_demo_main.c
+ │ ├── board.c
+ │ ├── board.h
+ │ ├── component.mk
+ │ └── Kconfig.projbuild
+ ├── Makefile
+ ├── README.md
+ ├── sdkconfig.defaults
+ └── tutorial
+ └── Ble_Mesh_Provisioner_Example_Walkthrough.md
+
+8 directories, 26 files
+```
+
+
+## Hardware and Setup
+
+At present ESP32-DevKitC and ESP-WROVER-KIT are supported for BLE Mesh implementation. You can find the details about the modules [here](https://docs.espressif.com/projects/esp-idf/en/latest/hw-reference/modules-and-boards.html)
+
+You can choose the board through menuconfig: `make menuconfig -> Example Configuration -> Board selection for BLE Mesh`
+
+Note that if you plan to use ESP32-DevKitC, you need to connect an RGB LED to GPIO pins 25, 26 and 27.
+
+
+## Sample Examples
+
+* BLE Mesh Node
+
+This example shows the use of BLE Mesh as a node device having a Configuration Server model and a Generic OnOff Server model. A BLE Mesh provisioner can then provision the node and control a RGB LED representing on/off state.
+
+* BLE Mesh Client Model
+
+This example shows how a Generic OnOff Client model within a node works. The node has a Configuration Server model, a Generic OnOff Server model and a Generic OnOff Client model.
+
+* BLE Mesh Provisioner
+
+This example shows how a device can act as a BLE Mesh provisioner to provision devices. The provisioner has a Configuration Server model, a Configuration Client model and a Generic OnOff Client model.
+
+
+## Mobile Apps
+
+ESP BLE Mesh implementation is compatible with a few phone apps, including Silicon Labs BLE Mesh and nRF Mesh. These apps are available on Google Play and App Store. In addition, Espressif offers its own Android app which is currently being actively developed. You can find the latest APK file [here](http://download.espressif.com/BLE_MESH/BLE_Mesh_Tools/BLE_Mesh_App/EspBleMesh-v0.9.2).
+
+Note: The most recent tested version 1.1.0 of Silicon Labs App has a bug, which has been fixed by a workaround on the SDK side. The fix is implemented through a configuration option enabled by default. For other Android/iOS apps, this option needs to be disabled from menuconfig:
+`make menuconfig -> Example Configuration -> This option fixes the bug of Silicon Lab Android App 1.1.0 when reconnection will cause the sequence number to recount from 0`
+
+## Building and Flashing
+
+If you build the application for the first time, the menuconfig screen will pop up. You can choose the board from the Example Configuration option. Additionally, you can modify the serial settings in the Serial flasher config option in accordance with your port configuration.
+
+BLE Mesh specific configuration options can also be modified through: `make menuconfig -> Component config -> Bluetooth Mesh support`
+
+You can still change options at any other time using `make menuconfig`.
+
+```
+$ export IDF_PATH=/path/to/esp-ble-mesh-sdk-v0.x
+
+$ cd examples/bluetooth/ble_mesh/
+
+$ make -j8 flash monitor
+```
+
+
+# Reporting Issues
+
+* If you find a bug or have a feature request, go to [the Issues section on GitHub](https://github.com/espressif/esp-idf/issues). Before reporting a new issue, please check the existing issues at the provided link and the FAQs document in the `mesh_docs` folder.
+* When you submit an issue or a feature request on GitHub, please add the tag "BLE Mesh" in the issue title for our faster reaction.
diff --git a/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Known_Issues_EN.md b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Known_Issues_EN.md
new file mode 100644
index 0000000000..034d2388d3
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_docs/BLE-Mesh_Known_Issues_EN.md
@@ -0,0 +1 @@
+To be added.
diff --git a/components/bt/ble_mesh/mesh_docs/README.md b/components/bt/ble_mesh/mesh_docs/README.md
new file mode 100644
index 0000000000..600fc3ac82
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_docs/README.md
@@ -0,0 +1,38 @@
+# ESP BLE Mesh Framework
+
+## Demos
+
+* [Provisioning of BLE Mesh nodes using Smartphone App](http://download.espressif.com/BLE_MESH/Docs4Customers/esp-ble-mesh-demo.mp4)
+* [Espressif Fast Provisioning using ESP BLE Mesh App](http://download.espressif.com/BLE_MESH/BLE_Mesh_Demo/V0.4_Demo_Fast_Provision/ESP32_BLE_Mesh_Fast_Provision.mp4)
+
+## Examples
+
+* [BLE Mesh Node Example Code](../../../../examples/bluetooth/ble_mesh/ble_mesh_node)
+* [BLE_Mesh_Node_Example_Walkthrough](../../../../examples/bluetooth/ble_mesh/ble_mesh_node/tutorial/Ble_Mesh_Node_Example_Walkthrough.md)
+* [BLE Mesh Provisioner Example Code](../../../../examples/bluetooth/ble_mesh/ble_mesh_provisioner)
+* [BLE_Mesh_Provisioner_Example_Walkthrough](../../../../examples/bluetooth/ble_mesh/ble_mesh_provisioner/tutorial/Ble_Mesh_Provisioner_Example_Walkthrough.md)
+* [BLE Mesh Client Model Example Code](../../../../examples/bluetooth/ble_mesh/ble_mesh_client_model)
+
+## Documentation
+
+### ESP BLE Mesh Development Documentation
+
+* [Getting started with ESP BLE Mesh](BLE-Mesh_Getting_Started_EN.md)
+* [ESP BLE Mesh Feature List](BLE-Mesh_Feature_List_EN.md)
+* [FAQs](BLE-Mesh_FAQs_EN.md)
+* [Known Issues](BLE-Mesh_Known_Issues_EN.md)
+
+### BLE Mesh Protocol Documentation
+
+* [BLE Mesh Core Specification](https://www.bluetooth.org/docman/handlers/downloaddoc.ashx?doc_id=429633)
+* [BLE Mesh Model Specification](https://www.bluetooth.org/docman/handlers/downloaddoc.ashx?doc_id=429634)
+* [An Intro to Bluetooth Mesh Part 1](http://blog.bluetooth.com/an-intro-to-bluetooth-mesh-part1)
+* [An Intro to Bluetooth Mesh Part 2](http://blog.bluetooth.com/an-intro-to-bluetooth-mesh-part2)
+* [The Fundamental Concepts of Bluetooth Mesh Networking Part 1](http://blog.bluetooth.com/the-fundamental-concepts-of-bluetooth-mesh-networking-part-1)
+* [The Fundamental Concepts of Bluetooth Mesh Networking, Part 2](http://blog.bluetooth.com/the-fundamental-concepts-of-bluetooth-mesh-networking-part-2)
+* [Bluetooth Mesh Networking: Friendship](http://blog.bluetooth.com/bluetooth-mesh-networking-series-friendship)
+* [Management of Devices in a Bluetooth Mesh Network](http://blog.bluetooth.com/management-of-devices-bluetooth-mesh-network)
+* [Bluetooth Mesh Security Overview](http://blog.bluetooth.com/bluetooth-mesh-security-overview)
+* [Provisioning a Bluetooth Mesh Network Part 1](http://blog.bluetooth.com/provisioning-a-bluetooth-mesh-network-part-1)
+* [Provisioning a Bluetooth Mesh Network Part 2](http://blog.bluetooth.com/provisioning-a-bluetooth-mesh-network-part-2)
+
diff --git a/components/bt/ble_mesh/mesh_models/bt_mesh_client_common.c b/components/bt/ble_mesh/mesh_models/bt_mesh_client_common.c
new file mode 100644
index 0000000000..1c0978048c
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/bt_mesh_client_common.c
@@ -0,0 +1,271 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "bt_mesh_client_common.h"
+#include "mesh_access.h"
+#include "mesh_buf.h"
+#include "mesh.h"
+#include "mesh_slist.h"
+#include "errno.h"
+#include "osi/allocator.h"
+#include
+
+bt_mesh_client_node_t *bt_mesh_is_model_message_publish(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf,
+ bool need_pub)
+{
+ bt_mesh_client_common_t *cli = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ bt_mesh_internal_data_t *data = NULL;
+ u32_t rsp;
+
+ if (!model || !ctx || !buf) {
+ BT_ERR("%s: invalid parameters", __func__);
+ return NULL;
+ }
+
+ cli = (bt_mesh_client_common_t *)model->user_data;
+ if (!cli) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return NULL;
+ }
+
+ rsp = ctx->recv_op;
+
+ /** If the received message address is not a unicast address,
+ * the address may be a group/virtual address, and we push
+ * this message to the application layer.
+ */
+ if (!BT_MESH_ADDR_IS_UNICAST(ctx->recv_dst)) {
+ BT_DBG("Unexpected status message 0x%x", rsp);
+ if (cli->publish_status && need_pub) {
+ cli->publish_status(rsp, model, ctx, buf);
+ }
+ return NULL;
+ }
+
+ /** If the source address of the received status message is
+ * different with the destination address of the sending
+ * message, then the message is from another element and
+ * push it to application layer.
+ */
+ data = (bt_mesh_internal_data_t *)cli->internal_data;
+ if (!data) {
+ BT_ERR("%s: internal_data is NULL", __func__);
+ return NULL;
+ }
+
+ if ((node = bt_mesh_client_pick_node(&data->queue, ctx->addr)) == NULL) {
+ BT_DBG("Unexpected status message 0x%x", rsp);
+ if (cli->publish_status && need_pub) {
+ cli->publish_status(rsp, model, ctx, buf);
+ }
+ return NULL;
+ }
+
+ if (node->op_pending != rsp) {
+ BT_DBG("Unexpected status message 0x%x", rsp);
+ if (cli->publish_status && need_pub) {
+ cli->publish_status(rsp, model, ctx, buf);
+ }
+ return NULL;
+ }
+
+ return node;
+}
+
+bool bt_mesh_client_find_opcode_in_list(sys_slist_t *list, u32_t opcode)
+{
+ if (sys_slist_is_empty(list)) {
+ return false;
+ }
+
+ sys_snode_t *cur = NULL; bt_mesh_client_node_t *node = NULL;
+ for (cur = sys_slist_peek_head(list);
+ cur != NULL; cur = sys_slist_peek_next(cur)) {
+ node = (bt_mesh_client_node_t *)cur;
+ if (node->op_pending == opcode) {
+ return true;
+ }
+ return NULL;
+ }
+
+ return node;
+}
+
+bool bt_mesh_client_check_node_in_list(sys_slist_t *list, u16_t tx_dst)
+{
+ if (sys_slist_is_empty(list)) {
+ return false;
+ }
+
+ sys_snode_t *cur = NULL; bt_mesh_client_node_t *node = NULL;
+ for (cur = sys_slist_peek_head(list);
+ cur != NULL; cur = sys_slist_peek_next(cur)) {
+ node = (bt_mesh_client_node_t *)cur;
+ if (node->ctx.addr == tx_dst) {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+bt_mesh_client_node_t *bt_mesh_client_pick_node(sys_slist_t *list, u16_t tx_dst)
+{
+ if (sys_slist_is_empty(list)) {
+ return NULL;
+ }
+
+ sys_snode_t *cur = NULL; bt_mesh_client_node_t *node = NULL;
+ for (cur = sys_slist_peek_head(list);
+ cur != NULL; cur = sys_slist_peek_next(cur)) {
+ node = (bt_mesh_client_node_t *)cur;
+ if (node->ctx.addr == tx_dst) {
+ return node;
+ }
+ }
+
+ return NULL;
+}
+
+static u32_t bt_mesh_client_get_status_op(const bt_mesh_client_op_pair_t *op_pair,
+ int size, u32_t opcode)
+{
+ if (!op_pair || size == 0) {
+ return 0;
+ }
+
+ const bt_mesh_client_op_pair_t *op = op_pair;
+ for (int i = 0; i < size; i++) {
+ if (op->cli_op == opcode) {
+ return op->status_op;
+ }
+ op++;
+ }
+
+ return 0;
+}
+
+int bt_mesh_client_send_msg(struct bt_mesh_model *model,
+ u32_t opcode,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *msg,
+ k_work_handler_t timer_handler,
+ s32_t timeout, bool need_ack,
+ const struct bt_mesh_send_cb *cb,
+ void *cb_data)
+{
+ bt_mesh_client_common_t *cli = NULL;
+ bt_mesh_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ int err;
+
+ if (!model || !ctx || !msg) {
+ BT_ERR("%s: invalid parameters", __func__);
+ return -EINVAL;
+ }
+
+ cli = (bt_mesh_client_common_t *)model->user_data;
+ __ASSERT(cli, "Invalid client value when sent client msg.");
+ internal = (bt_mesh_internal_data_t *)cli->internal_data;
+ __ASSERT(internal, "Invalid internal value when sent client msg.");
+
+ if (!need_ack) {
+ /* If this is an unack message, send it directly. */
+ return bt_mesh_model_send(model, ctx, msg, cb, cb_data);
+ }
+
+ if (bt_mesh_client_check_node_in_list(&internal->queue, ctx->addr)) {
+ err = -EBUSY;
+ } else {
+ /* Don't forget to free the node in the timeout (timer_handler) function. */
+ node = (bt_mesh_client_node_t *)osi_calloc(sizeof(bt_mesh_client_node_t));
+ if (!node) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return -ENOMEM;
+ }
+ memcpy(&node->ctx, ctx, sizeof(struct bt_mesh_msg_ctx));
+ node->ctx.model = model;
+ node->opcode = opcode;
+ if ((node->op_pending = bt_mesh_client_get_status_op(cli->op_pair, cli->op_pair_size, opcode)) == 0) {
+ BT_WARN("%s, Didn't find the status opcode in the op_pair list.", __func__);
+ osi_free(node);
+ return -EINVAL;
+ }
+ if ((err = bt_mesh_model_send(model, ctx, msg, cb, cb_data)) != 0) {
+ osi_free(node);
+ } else {
+ sys_slist_append(&internal->queue, &node->client_node);
+ k_delayed_work_init(&node->timer, timer_handler);
+ k_delayed_work_submit(&node->timer, timeout ? timeout : CONFIG_BT_MESH_CLIENT_MSG_TIMEOUT);
+ }
+ }
+
+ return err;
+}
+
+int bt_mesh_client_init(struct bt_mesh_model *model)
+{
+ bt_mesh_client_common_t *cli = NULL;
+ bt_mesh_internal_data_t *data = NULL;
+
+ if (!model) {
+ BT_ERR("Client model is NULL");
+ return -EINVAL;
+ }
+
+ if (!model->op) {
+ BT_ERR("Client model op is NULL");
+ return -EINVAL;
+ }
+
+ cli = model->user_data;
+ if (!cli) {
+ BT_ERR("Client model user_data is NULL");
+ return -EINVAL;
+ }
+
+ /* TODO: call osi_free() when deinit function is invoked */
+ data = osi_calloc(sizeof(bt_mesh_internal_data_t));
+ if (!data) {
+ BT_ERR("Allocate memory for client internal data fail");
+ return -ENOMEM;
+ }
+
+ /* Init the client data queue */
+ sys_slist_init(&data->queue);
+
+ cli->model = model;
+ cli->internal_data = data;
+
+ return 0;
+}
+
+int bt_mesh_client_free_node(sys_slist_t *queue, bt_mesh_client_node_t *node)
+{
+ if (!queue || !node) {
+ return -EINVAL;
+ }
+
+ // Free the node timer
+ k_delayed_work_free(&node->timer);
+ // Release the client node from the queue
+ sys_slist_find_and_remove(queue, &node->client_node);
+ // Free the node
+ osi_free(node);
+
+ return 0;
+}
diff --git a/components/bt/ble_mesh/mesh_models/common.c b/components/bt/ble_mesh/mesh_models/common.c
new file mode 100644
index 0000000000..709a7da953
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/common.c
@@ -0,0 +1,110 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "osi/allocator.h"
+#include "errno.h"
+
+#include "mesh_types.h"
+#include "mesh_kernel.h"
+#include "mesh_buf.h"
+#include "mesh_trace.h"
+#include "common.h"
+#include "bt_mesh_client_common.h"
+
+struct net_buf_simple *bt_mesh_alloc_buf(u16_t size)
+{
+ struct net_buf_simple *buf;
+
+ buf = (struct net_buf_simple *)osi_calloc(sizeof(struct net_buf_simple) + size);
+ if (!buf) {
+ BT_ERR("%s: Failed to allocate memory", __func__);
+ return NULL;
+ }
+
+ /* buf->data will be initialized in net_buf_simple_init() */
+ buf->len = 0;
+ buf->size = size;
+
+ return buf;
+}
+
+void bt_mesh_free_buf(struct net_buf_simple *buf)
+{
+ if (buf) {
+ osi_free(buf);
+ }
+}
+
+int bt_mesh_copy_msg_role(bt_mesh_role_param_t *common)
+{
+ bt_mesh_client_common_t *client = NULL;
+
+ if (!common || !common->model || !common->model->user_data) {
+ BT_ERR("%s: invalid parameter", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_client_common_t *)common->model->user_data;
+
+ switch (common->role) {
+#if CONFIG_BT_MESH_NODE
+ case NODE:
+ /* no matter if provisioner is enabled/disabled , node role can be used to send messages */
+ client->msg_role = NODE;
+ break;
+#endif
+#if CONFIG_BT_MESH_PROVISIONER
+ case PROVISIONER:
+ /* if provisioner is not enabled, provisioner role can't be used to send messages */
+ if (!bt_mesh_is_provisioner_en()) {
+ BT_WARN("%s: send provisioner msg with provisioner disabled", __func__);
+ return -EINVAL;
+ }
+ client->msg_role = PROVISIONER;
+ break;
+#endif
+#if CONFIG_BT_MESH_FAST_PROV
+ case FAST_PROV:
+ client->msg_role = FAST_PROV;
+ break;
+#endif
+ default:
+ BT_WARN("%s: role mismatch", __func__);
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+u8_t bt_mesh_get_msg_role(struct bt_mesh_model *model, bool srv_send)
+{
+ bt_mesh_client_common_t *client = NULL;
+
+ if (srv_send) {
+ BT_DBG("%s: msg sent by a server model", __func__);
+ return NODE;
+ }
+
+ if (!model || !model->user_data) {
+ BT_ERR("%s: invalid parameter", __func__);
+ return ROLE_NVAL;
+ }
+
+ client = (bt_mesh_client_common_t *)model->user_data;
+
+ return client->msg_role;
+}
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_models/generic_client.c b/components/bt/ble_mesh/mesh_models/generic_client.c
new file mode 100644
index 0000000000..ce6ab135af
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/generic_client.c
@@ -0,0 +1,1233 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+#include
+
+#include "osi/allocator.h"
+
+#include "mesh_types.h"
+#include "mesh_kernel.h"
+#include "mesh_trace.h"
+#include "mesh.h"
+
+#include "model_op.h"
+#include "common.h"
+#include "generic_client.h"
+
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_generic_client.h"
+
+/** The following are the macro definitions of generic client
+ * model messages length, and a message is composed of three
+ * parts: Opcode + msg_value + MIC
+ */
+/* Generic onoff client messages length */
+#define BT_MESH_GEN_ONOFF_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_ONOFF_SET_MSG_LEN (2 + 4 + 4)
+
+/* Generic level client messages length */
+#define BT_MESH_GEN_LEVEL_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_LEVEL_SET_MSG_LEN (2 + 5 + 4)
+#define BT_MESH_GEN_DELTA_SET_MSG_LEN (2 + 7 + 4)
+#define BT_MESH_GEN_MOVE_SET_MSG_LEN (2 + 5 + 4)
+
+/* Generic default transition time client messages length */
+#define BT_MESH_GEN_DEF_TRANS_TIME_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_DEF_TRANS_TIME_SET_MSG_LEN (2 + 1 + 4)
+
+/* Generic power onoff client messages length */
+#define BT_MESH_GEN_ONPOWERUP_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_ONPOWERUP_SET_MSG_LEN (2 + 1 + 4)
+
+/* Generic power level client messages length */
+#define BT_MESH_GEN_POWER_LEVEL_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_POWER_LEVEL_SET_MSG_LEN (2 + 5 + 4)
+#define BT_MESH_GEN_POWER_LAST_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_POWER_DEFAULT_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_POWER_DEFAULT_SET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_GEN_POWER_RANGE_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_POWER_RANGE_SET_MSG_LEN (2 + 4 + 4)
+
+/* Generic battery client messages length */
+#define BT_MESH_GEN_BATTERY_GET_MSG_LEN (2 + 0 + 4)
+
+/* Generic location client messages length */
+#define BT_MESH_GEN_LOC_GLOBAL_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_LOC_GLOBAL_SET_MSG_LEN (1 + 10 + 4)
+#define BT_MESH_GEN_LOC_LOCAL_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_LOC_LOCAL_SET_MSG_LEN (2 + 9 + 4)
+
+/* Generic property client messages length */
+#define BT_MESH_GEN_USER_PROPERTIES_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_USER_PROPERTY_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_GEN_USER_PROPERTY_SET_MSG_LEN /* variable */
+#define BT_MESH_GEN_ADMIN_PROPERTIES_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_ADMIN_PROPERTY_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_GEN_ADMIN_PROPERTY_SET_MSG_LEN /* variable */
+#define BT_MESH_GEN_MANU_PROPERTIES_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_GEN_MANU_PROPERTY_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_GEN_MANU_PROPERTY_SET_MSG_LEN (1 + 3 + 4)
+#define BT_MESH_GEN_CLINET_PROPERTIES_GET_MSG_LEN (1 + 2 + 4)
+
+#define BT_MESH_GEN_GET_STATE_MSG_LEN (2 + 2 + 4)
+
+static const bt_mesh_client_op_pair_t gen_op_pair[] = {
+ { BT_MESH_MODEL_OP_GEN_ONOFF_GET, BT_MESH_MODEL_OP_GEN_ONOFF_STATUS },
+ { BT_MESH_MODEL_OP_GEN_ONOFF_SET, BT_MESH_MODEL_OP_GEN_ONOFF_STATUS },
+ { BT_MESH_MODEL_OP_GEN_LEVEL_GET, BT_MESH_MODEL_OP_GEN_LEVEL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_LEVEL_SET, BT_MESH_MODEL_OP_GEN_LEVEL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_DELTA_SET, BT_MESH_MODEL_OP_GEN_LEVEL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_MOVE_SET, BT_MESH_MODEL_OP_GEN_LEVEL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_GET, BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS },
+ { BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET, BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS },
+ { BT_MESH_MODEL_OP_GEN_ONPOWERUP_GET, BT_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS },
+ { BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET, BT_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS },
+ { BT_MESH_MODEL_OP_GEN_POWER_LEVEL_GET, BT_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET, BT_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_POWER_LAST_GET, BT_MESH_MODEL_OP_GEN_POWER_LAST_STATUS },
+ { BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_GET, BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET, BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_GEN_POWER_RANGE_GET, BT_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET, BT_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_GEN_BATTERY_GET, BT_MESH_MODEL_OP_GEN_BATTERY_STATUS },
+ { BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_GET, BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET, BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_LOC_LOCAL_GET, BT_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET, BT_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS },
+ { BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET, BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS },
+ { BT_MESH_MODEL_OP_GEN_USER_PROPERTY_GET, BT_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS },
+ { BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET, BT_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS },
+ { BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET, BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS },
+ { BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET, BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS },
+ { BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET, BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS },
+ { BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_GET, BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_STATUS },
+ { BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_GET, BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_STATUS },
+ { BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET, BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_STATUS },
+ { BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET, BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS },
+};
+
+static void timeout_handler(struct k_work *work)
+{
+ bt_mesh_generic_client_t *client = NULL;
+ generic_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+
+ BT_WARN("Receive generic status message timeout");
+
+ node = CONTAINER_OF(work, bt_mesh_client_node_t, timer.work);
+ if (!node || !node->ctx.model) {
+ BT_ERR("%s: node parameter is NULL", __func__);
+ return;
+ }
+
+ client = (bt_mesh_generic_client_t *)node->ctx.model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (generic_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: internal_data is NULL", __func__);
+ return;
+ }
+
+ bt_mesh_callback_generic_status_to_btc(node->opcode, 0x03, node->ctx.model,
+ &node->ctx, NULL, 0);
+
+ bt_mesh_client_free_node(&internal->queue, node);
+
+ return;
+}
+
+static void generic_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ bt_mesh_generic_client_t *client = NULL;
+ generic_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ u8_t *val = NULL;
+ u8_t evt = 0xFF;
+ u32_t rsp = 0;
+ size_t len = 0;
+
+ BT_DBG("%s: len %d, bytes %s", __func__, buf->len, bt_hex(buf->data, buf->len));
+
+ client = (bt_mesh_generic_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (generic_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: model internal_data is NULL", __func__);
+ return;
+ }
+
+ rsp = ctx->recv_op;
+
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_GEN_ONOFF_STATUS: {
+ struct bt_mesh_gen_onoff_status *status = NULL;
+ if (buf->len != 1 && buf->len != 3) {
+ BT_ERR("Wrong Generic OnOff status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_onoff_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_onoff = net_buf_simple_pull_u8(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_onoff = net_buf_simple_pull_u8(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_onoff_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_LEVEL_STATUS: {
+ struct bt_mesh_gen_level_status *status = NULL;
+ if (buf->len != 2 && buf->len != 5) {
+ BT_ERR("Wrong Generic level status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_level_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_level = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_level = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_level_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS: {
+ struct bt_mesh_gen_def_trans_time_status *status = NULL;
+ if (buf->len != 1) {
+ BT_ERR("Wrong Generic default transition time status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_def_trans_time_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->trans_time = net_buf_simple_pull_u8(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_def_trans_time_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS: {
+ struct bt_mesh_gen_onpowerup_status *status = NULL;
+ if (buf->len != 1) {
+ BT_ERR("Wrong Generic OnPowerUp status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_onpowerup_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->onpowerup = net_buf_simple_pull_u8(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_onpowerup_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS: {
+ struct bt_mesh_gen_power_level_status *status = NULL;
+ if (buf->len != 2 && buf->len != 5) {
+ BT_ERR("Wrong Generic power level status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_power_level_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_power = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_power = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_power_level_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_POWER_LAST_STATUS: {
+ struct bt_mesh_gen_power_last_status *status = NULL;
+ if (buf->len != 2) {
+ BT_ERR("Wrong Generic power last status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_power_last_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->power = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_power_last_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS: {
+ struct bt_mesh_gen_power_default_status *status = NULL;
+ if (buf->len != 2) {
+ BT_ERR("Wrong Generic power default status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_power_default_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->power = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_power_default_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS: {
+ struct bt_mesh_gen_power_range_status *status = NULL;
+ if (buf->len != 5) {
+ BT_ERR("Wrong Generic power range status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_power_range_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->status_code = net_buf_simple_pull_u8(buf);
+ status->range_min = net_buf_simple_pull_le16(buf);
+ status->range_max = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_power_range_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_BATTERY_STATUS: {
+ struct bt_mesh_gen_battery_status *status = NULL;
+ if (buf->len != 8) {
+ BT_ERR("Wrong Generic battery status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_battery_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ u32_t value = 0;
+ value = net_buf_simple_pull_le32(buf);
+ status->battery_level = (u8_t)value;
+ status->time_to_discharge = (value >> 8);
+ value = net_buf_simple_pull_le32(buf);
+ status->time_to_charge = (value & 0xffffff);
+ status->flags = (u8_t)(value >> 24);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_battery_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS: {
+ struct bt_mesh_gen_loc_global_status *status = NULL;
+ if (buf->len != 10) {
+ BT_ERR("Wrong Generic location global status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_loc_global_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->global_latitude = net_buf_simple_pull_le32(buf);
+ status->global_longitude = net_buf_simple_pull_le32(buf);
+ status->global_altitude = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_loc_global_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS: {
+ struct bt_mesh_gen_loc_local_status *status = NULL;
+ if (buf->len != 9) {
+ BT_ERR("Wrong Generic location local status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_gen_loc_local_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->local_north = net_buf_simple_pull_le16(buf);
+ status->local_east = net_buf_simple_pull_le16(buf);
+ status->local_altitude = net_buf_simple_pull_le16(buf);
+ status->floor_number = net_buf_simple_pull_u8(buf);
+ status->uncertainty = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_loc_local_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_user_properties_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_gen_user_properties_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->user_property_ids = bt_mesh_alloc_buf(buf->len);
+ if (!status->user_property_ids) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->user_property_ids, 0);
+ net_buf_simple_add_mem(status->user_property_ids, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_user_properties_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS: {
+ struct bt_mesh_gen_user_property_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_gen_user_property_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->user_property_id = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->user_access = net_buf_simple_pull_u8(buf);
+ status->user_property_value = bt_mesh_alloc_buf(buf->len);
+ if (!status->user_property_value) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->user_property_value, 0);
+ net_buf_simple_add_mem(status->user_property_value, buf->data, buf->len);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_user_property_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_admin_properties_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_gen_admin_properties_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->admin_property_ids = bt_mesh_alloc_buf(buf->len);
+ if (!status->admin_property_ids) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->admin_property_ids, 0);
+ net_buf_simple_add_mem(status->admin_property_ids, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_admin_properties_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS: {
+ struct bt_mesh_gen_admin_property_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_gen_admin_property_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->admin_property_id = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->admin_user_access = net_buf_simple_pull_u8(buf);
+ status->admin_property_value = bt_mesh_alloc_buf(buf->len);
+ if (!status->admin_property_value) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->admin_property_value, 0);
+ net_buf_simple_add_mem(status->admin_property_value, buf->data, buf->len);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_admin_property_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_manu_properties_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_gen_manu_properties_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->manu_property_ids = bt_mesh_alloc_buf(buf->len);
+ if (!status->manu_property_ids) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->manu_property_ids, 0);
+ net_buf_simple_add_mem(status->manu_property_ids, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_manu_properties_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_STATUS: {
+ struct bt_mesh_gen_manu_property_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_gen_manu_property_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->manu_property_id = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->manu_user_access = net_buf_simple_pull_u8(buf);
+ status->manu_property_value = bt_mesh_alloc_buf(buf->len);
+ if (!status->manu_property_value) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->manu_property_value, 0);
+ net_buf_simple_add_mem(status->manu_property_value, buf->data, buf->len);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_manu_property_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_client_properties_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_gen_client_properties_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->client_property_ids = bt_mesh_alloc_buf(buf->len);
+ if (!status->client_property_ids) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->client_property_ids, 0);
+ net_buf_simple_add_mem(status->client_property_ids, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_gen_client_properties_status);
+ break;
+ }
+ default:
+ BT_ERR("Not a generic status message opcode");
+ return;
+ }
+
+ buf->data = val;
+ buf->len = len;
+ node = bt_mesh_is_model_message_publish(model, ctx, buf, true);
+ if (!node) {
+ BT_DBG("Unexpected generic status message 0x%x", rsp);
+ } else {
+ switch (node->opcode) {
+ case BT_MESH_MODEL_OP_GEN_ONOFF_GET:
+ case BT_MESH_MODEL_OP_GEN_LEVEL_GET:
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_GET:
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_LAST_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_GET:
+ case BT_MESH_MODEL_OP_GEN_BATTERY_GET:
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_GET:
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_GET:
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET:
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_GET:
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET:
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET:
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_GET:
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_GET:
+ case BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET:
+ evt = 0x00;
+ break;
+ case BT_MESH_MODEL_OP_GEN_ONOFF_SET:
+ case BT_MESH_MODEL_OP_GEN_LEVEL_SET:
+ case BT_MESH_MODEL_OP_GEN_DELTA_SET:
+ case BT_MESH_MODEL_OP_GEN_MOVE_SET:
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET:
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET:
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET:
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET:
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET:
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET:
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET:
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET:
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET:
+ evt = 0x01;
+ break;
+ default:
+ break;
+ }
+
+ bt_mesh_callback_generic_status_to_btc(node->opcode, evt, model, ctx, val, len);
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&internal->queue, node);
+ }
+
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_user_properties_status *status;
+ status = (struct bt_mesh_gen_user_properties_status *)val;
+ bt_mesh_free_buf(status->user_property_ids);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS: {
+ struct bt_mesh_gen_user_property_status *status;
+ status = (struct bt_mesh_gen_user_property_status *)val;
+ bt_mesh_free_buf(status->user_property_value);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_admin_properties_status *status;
+ status = (struct bt_mesh_gen_admin_properties_status *)val;
+ bt_mesh_free_buf(status->admin_property_ids);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS: {
+ struct bt_mesh_gen_admin_property_status *status;
+ status = (struct bt_mesh_gen_admin_property_status *)val;
+ bt_mesh_free_buf(status->admin_property_value);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_manu_properties_status *status;
+ status = (struct bt_mesh_gen_manu_properties_status *)val;
+ bt_mesh_free_buf(status->manu_property_ids);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_STATUS: {
+ struct bt_mesh_gen_manu_property_status *status;
+ status = (struct bt_mesh_gen_manu_property_status *)val;
+ bt_mesh_free_buf(status->manu_property_value);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS: {
+ struct bt_mesh_gen_client_properties_status *status;
+ status = (struct bt_mesh_gen_client_properties_status *)val;
+ bt_mesh_free_buf(status->client_property_ids);
+ break;
+ }
+ default:
+ break;
+ }
+
+ osi_free(val);
+
+ return;
+}
+
+const struct bt_mesh_model_op gen_onoff_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_ONOFF_STATUS, 1, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op gen_level_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_LEVEL_STATUS, 2, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op gen_def_trans_time_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS, 1, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op gen_power_onoff_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS, 1, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op gen_power_level_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_POWER_LAST_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS, 5, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op gen_battery_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_BATTERY_STATUS, 8, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op gen_location_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS, 10, generic_status },
+ { BT_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS, 9, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op gen_property_cli_op[] = {
+ { BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_STATUS, 2, generic_status },
+ { BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS, 2, generic_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+static int gen_get_state(struct bt_mesh_common_param *common, void *value)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(BT_MESH_GEN_GET_STATE_MSG_LEN);
+ int err;
+
+ bt_mesh_model_msg_init(msg, common->opcode);
+
+ if (value) {
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_GET: {
+ struct bt_mesh_gen_user_property_get *get;
+ get = (struct bt_mesh_gen_user_property_get *)value;
+ net_buf_simple_add_le16(msg, get->user_property_id);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET: {
+ struct bt_mesh_gen_admin_property_get *get;
+ get = (struct bt_mesh_gen_admin_property_get *)value;
+ net_buf_simple_add_le16(msg, get->admin_property_id);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_GET: {
+ struct bt_mesh_gen_manu_property_get *get;
+ get = (struct bt_mesh_gen_manu_property_get *)value;
+ net_buf_simple_add_le16(msg, get->manu_property_id);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET: {
+ struct bt_mesh_gen_client_properties_get *get;
+ get = (struct bt_mesh_gen_client_properties_get *)value;
+ net_buf_simple_add_le16(msg, get->client_property_id);
+ break;
+ }
+ default:
+ BT_DBG("This generic message should be sent with NULL get pointer");
+ break;
+ }
+ }
+
+ err = bt_mesh_client_send_msg(common->model, common->opcode, &common->ctx, msg,
+ timeout_handler, common->msg_timeout, true,
+ common->cb, common->cb_data);
+ if (err) {
+ BT_ERR("Generic get message send failed (err %d)", err);
+ }
+
+ return err;
+}
+
+static int gen_set_state(struct bt_mesh_common_param *common,
+ void *value, u16_t value_len, bool need_ack)
+{
+ struct net_buf_simple *msg = NULL;
+ int err;
+
+ msg = bt_mesh_alloc_buf(value_len);
+ if (!msg) {
+ BT_ERR("Generic set allocate memory fail");
+ return -ENOMEM;
+ }
+
+ bt_mesh_model_msg_init(msg, common->opcode);
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_GEN_ONOFF_SET:
+ case BT_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK: {
+ struct bt_mesh_gen_onoff_set *set;
+ set = (struct bt_mesh_gen_onoff_set *)value;
+ net_buf_simple_add_u8(msg, set->onoff);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_LEVEL_SET:
+ case BT_MESH_MODEL_OP_GEN_LEVEL_SET_UNACK: {
+ struct bt_mesh_gen_level_set *set;
+ set = (struct bt_mesh_gen_level_set *)value;
+ net_buf_simple_add_le16(msg, set->level);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_DELTA_SET:
+ case BT_MESH_MODEL_OP_GEN_DELTA_SET_UNACK: {
+ struct bt_mesh_gen_delta_set *set;
+ set = (struct bt_mesh_gen_delta_set *)value;
+ net_buf_simple_add_le32(msg, set->level);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_MOVE_SET:
+ case BT_MESH_MODEL_OP_GEN_MOVE_SET_UNACK: {
+ struct bt_mesh_gen_move_set *set;
+ set = (struct bt_mesh_gen_move_set *)value;
+ net_buf_simple_add_le16(msg, set->delta_level);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET:
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET_UNACK: {
+ struct bt_mesh_gen_def_trans_time_set *set;
+ set = (struct bt_mesh_gen_def_trans_time_set *)value;
+ net_buf_simple_add_u8(msg, set->trans_time);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET:
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET_UNACK: {
+ struct bt_mesh_gen_onpowerup_set *set;
+ set = (struct bt_mesh_gen_onpowerup_set *)value;
+ net_buf_simple_add_u8(msg, set->onpowerup);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET:
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET_UNACK: {
+ struct bt_mesh_gen_power_level_set *set;
+ set = (struct bt_mesh_gen_power_level_set *)value;
+ net_buf_simple_add_le16(msg, set->power);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET_UNACK: {
+ struct bt_mesh_gen_power_default_set *set;
+ set = (struct bt_mesh_gen_power_default_set *)value;
+ net_buf_simple_add_le16(msg, set->power);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET:
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET_UNACK: {
+ struct bt_mesh_gen_power_range_set *set;
+ set = (struct bt_mesh_gen_power_range_set *)value;
+ net_buf_simple_add_le16(msg, set->range_min);
+ net_buf_simple_add_le16(msg, set->range_max);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET:
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET_UNACK: {
+ struct bt_mesh_gen_loc_global_set *set;
+ set = (struct bt_mesh_gen_loc_global_set *)value;
+ net_buf_simple_add_le32(msg, set->global_latitude);
+ net_buf_simple_add_le32(msg, set->global_longitude);
+ net_buf_simple_add_le16(msg, set->global_altitude);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET:
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET_UNACK: {
+ struct bt_mesh_gen_loc_local_set *set;
+ set = (struct bt_mesh_gen_loc_local_set *)value;
+ net_buf_simple_add_le16(msg, set->local_north);
+ net_buf_simple_add_le16(msg, set->local_east);
+ net_buf_simple_add_le16(msg, set->local_altitude);
+ net_buf_simple_add_u8(msg, set->floor_number);
+ net_buf_simple_add_le16(msg, set->uncertainty);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET:
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET_UNACK: {
+ struct bt_mesh_gen_user_property_set *set;
+ set = (struct bt_mesh_gen_user_property_set *)value;
+ net_buf_simple_add_le16(msg, set->user_property_id);
+ net_buf_simple_add_mem(msg, set->user_property_value->data, set->user_property_value->len);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET:
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET_UNACK: {
+ struct bt_mesh_gen_admin_property_set *set;
+ set = (struct bt_mesh_gen_admin_property_set *)value;
+ net_buf_simple_add_le16(msg, set->admin_property_id);
+ net_buf_simple_add_u8(msg, set->admin_user_access);
+ net_buf_simple_add_mem(msg, set->admin_property_value->data, set->admin_property_value->len);
+ break;
+ }
+
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET:
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET_UNACK: {
+ struct bt_mesh_gen_manu_property_set *set;
+ set = (struct bt_mesh_gen_manu_property_set *)value;
+ net_buf_simple_add_le16(msg, set->manu_property_id);
+ net_buf_simple_add_u8(msg, set->manu_user_access);
+ break;
+ }
+
+ default:
+ BT_ERR("Not a generic client model set message opcode");
+ err = -EINVAL;
+ goto end;
+ }
+
+ err = bt_mesh_client_send_msg(common->model, common->opcode, &common->ctx, msg,
+ timeout_handler, common->msg_timeout, need_ack,
+ common->cb, common->cb_data);
+ if (err) {
+ BT_ERR("Generic set message send failed (err %d)", err);
+ }
+
+end:
+ bt_mesh_free_buf(msg);
+
+ return err;
+}
+
+int bt_mesh_generic_client_get_state(struct bt_mesh_common_param *common, void *get, void *status)
+{
+ bt_mesh_generic_client_t *client = NULL;
+
+ if (!common || !common->model) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_generic_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_GEN_ONOFF_GET:
+ case BT_MESH_MODEL_OP_GEN_LEVEL_GET:
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_GET:
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_LAST_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_GET:
+ case BT_MESH_MODEL_OP_GEN_BATTERY_GET:
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_GET:
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_GET:
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET:
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET:
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_GET:
+ break;
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_GET:
+ if (!get) {
+ BT_ERR("Generic user_property_get is NULL");
+ return -EINVAL;
+ }
+ break;
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET:
+ if (!get) {
+ BT_ERR("Generic admin_property_get is NULL");
+ return -EINVAL;
+ }
+ break;
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_GET:
+ if (!get) {
+ BT_ERR("Generic manu_property_get is NULL");
+ return -EINVAL;
+ }
+ break;
+ case BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET:
+ if (!get) {
+ BT_ERR("Generic client_properties_get is NULL");
+ return -EINVAL;
+ }
+ break;
+ default:
+ BT_ERR("Not a generic get message opcode");
+ return -EINVAL;
+ }
+
+ return gen_get_state(common, get);
+}
+
+int bt_mesh_generic_client_set_state(struct bt_mesh_common_param *common, void *set, void *status)
+{
+ bt_mesh_generic_client_t *client = NULL;
+ u16_t length = 0;
+ bool need_ack = false;
+
+ if (!common || !common->model || !set) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_generic_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_GEN_ONOFF_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK: {
+ struct bt_mesh_gen_onoff_set *value;
+ value = (struct bt_mesh_gen_onoff_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Generic onoff set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_GEN_ONOFF_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_LEVEL_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_LEVEL_SET_UNACK: {
+ struct bt_mesh_gen_level_set *value;
+ value = (struct bt_mesh_gen_level_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Generic level set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_GEN_LEVEL_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_DELTA_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_DELTA_SET_UNACK: {
+ struct bt_mesh_gen_delta_set *value;
+ value = (struct bt_mesh_gen_delta_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Generic delta set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_GEN_DELTA_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_MOVE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_MOVE_SET_UNACK: {
+ struct bt_mesh_gen_move_set *value;
+ value = (struct bt_mesh_gen_move_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Generic move set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_GEN_MOVE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET_UNACK: {
+ u8_t value = *(u8_t *)set;
+ if ((value & 0x3F) > 0x3E) {
+ BT_ERR("Generic default transition time set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ length = BT_MESH_GEN_DEF_TRANS_TIME_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET_UNACK:
+ length = BT_MESH_GEN_ONPOWERUP_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET_UNACK: {
+ struct bt_mesh_gen_power_level_set *value;
+ value = (struct bt_mesh_gen_power_level_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Generic power level set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_GEN_POWER_LEVEL_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET_UNACK:
+ length = BT_MESH_GEN_POWER_DEFAULT_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET_UNACK: {
+ struct bt_mesh_gen_power_range_set *value;
+ value = (struct bt_mesh_gen_power_range_set *)set;
+ if (value->range_min > value->range_max) {
+ BT_ERR("Generic power level set range min is greater than range max");
+ return -EINVAL;
+ }
+ length = BT_MESH_GEN_POWER_RANGE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET_UNACK:
+ length = BT_MESH_GEN_LOC_GLOBAL_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET_UNACK:
+ length = BT_MESH_GEN_LOC_LOCAL_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET_UNACK: {
+ struct bt_mesh_gen_user_property_set *value;
+ value = (struct bt_mesh_gen_user_property_set *)set;
+ if (!value->user_property_value) {
+ BT_ERR("Generic user_property_value is NULL");
+ return -EINVAL;
+ }
+ length = (1 + 2 + value->user_property_value->len + 4);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET_UNACK: {
+ struct bt_mesh_gen_admin_property_set *value;
+ value = (struct bt_mesh_gen_admin_property_set *)set;
+ if (!value->admin_property_value) {
+ BT_ERR("Generic admin_property_value is NULL");
+ return -EINVAL;
+ }
+ length = (1 + 2 + 1 + value->admin_property_value->len + 4);
+ break;
+ }
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET_UNACK:
+ length = BT_MESH_GEN_MANU_PROPERTY_SET_MSG_LEN;
+ break;
+ default:
+ BT_ERR("Not a generic set message opcode");
+ return -EINVAL;
+ }
+
+ return gen_set_state(common, set, length, need_ack);
+}
+
+static int generic_client_init(struct bt_mesh_model *model, bool primary)
+{
+ bt_mesh_generic_client_t *client = NULL;
+ generic_internal_data_t *internal = NULL;
+
+ BT_DBG("primary %u", primary);
+
+ if (!model) {
+ BT_ERR("Generic client model is NULL");
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_generic_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("Generic client model user_data is NULL");
+ return -EINVAL;
+ }
+
+ /* TODO: call osi_free() when deinit function is invoked*/
+ internal = osi_calloc(sizeof(generic_internal_data_t));
+ if (!internal) {
+ BT_ERR("Allocate memory for generic onoff internal data fail");
+ return -ENOMEM;
+ }
+
+ sys_slist_init(&internal->queue);
+
+ client->model = model;
+ client->op_pair_size = ARRAY_SIZE(gen_op_pair);
+ client->op_pair = gen_op_pair;
+ client->internal_data = internal;
+
+ return 0;
+}
+
+int bt_mesh_gen_onoff_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
+
+int bt_mesh_gen_level_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
+
+int bt_mesh_gen_def_trans_time_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
+
+int bt_mesh_gen_pwr_onoff_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
+
+int bt_mesh_gen_pwr_level_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
+
+int bt_mesh_gen_battery_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
+
+int bt_mesh_gen_location_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
+
+int bt_mesh_gen_property_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return generic_client_init(model, primary);
+}
diff --git a/components/bt/ble_mesh/mesh_models/include/bt_mesh_client_common.h b/components/bt/ble_mesh/mesh_models/include/bt_mesh_client_common.h
new file mode 100644
index 0000000000..cd68fa9892
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/include/bt_mesh_client_common.h
@@ -0,0 +1,92 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BT_MESH_CLIENT_COMMON_H_
+#define _BT_MESH_CLIENT_COMMON_H_
+
+#include "mesh_access.h"
+
+/** Mesh Client Model Context */
+typedef struct {
+ u32_t cli_op; /* The client opcode */
+ u32_t status_op; /* The server status opcode corresponding to the client opcode */
+} bt_mesh_client_op_pair_t;
+
+/** Mesh Client Model Context */
+typedef struct {
+ struct bt_mesh_model *model;
+ int op_pair_size; /* the size of op_pair */
+ const bt_mesh_client_op_pair_t *op_pair;
+ /**
+ * @brief This function is a callback function used to push the received unsolicited
+ * messages to the application layer.
+ *
+ * @param[in] opcode: Opcode of received status message
+ * @param[in] model: Model associated with the status message
+ * @param[in] ctx: Context information of the status message
+ * @param[in] buf: Buffer contains the status message value
+ *
+ * @return None
+ */
+ void (*publish_status)(u32_t opcode, struct bt_mesh_model *model, struct bt_mesh_msg_ctx *ctx, struct net_buf_simple *buf);
+ void *internal_data; /* Pointer of the structure of internal data */
+ u8_t msg_role; /* device role of the tx message */
+} bt_mesh_client_common_t;
+
+typedef struct {
+ sys_slist_t queue;
+} bt_mesh_internal_data_t;
+
+typedef struct {
+ sys_snode_t client_node;
+ struct bt_mesh_msg_ctx ctx;
+ u32_t opcode; /* Indicate the opcode of the message sending */
+ u32_t op_pending; /* Indicate the status message waiting for */
+ struct k_delayed_work timer; /* Message send Timer. Only for stack-internal use. */
+} bt_mesh_client_node_t;
+
+int bt_mesh_client_init(struct bt_mesh_model *model);
+
+/**
+ * @brief Check the msg is a publish msg or not
+ *
+ * @param model Mesh (client) Model that the message belongs to.
+ * @param ctx Message context, includes keys, TTL, etc.
+ * @param buf The message buffer
+ * @param need_pub Indicate if the msg sent to app layer as a publish msg
+ * @return 0 on success, or (negative) error code on failure.
+ */
+bt_mesh_client_node_t *bt_mesh_is_model_message_publish(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf,
+ bool need_pub);
+
+bool bt_mesh_client_find_opcode_in_list(sys_slist_t *list, u32_t opcode);
+
+bool bt_mesh_client_check_node_in_list(sys_slist_t *list, uint16_t tx_dst);
+
+bt_mesh_client_node_t *bt_mesh_client_pick_node(sys_slist_t *list, u16_t tx_dst);
+
+int bt_mesh_client_send_msg(struct bt_mesh_model *model,
+ u32_t opcode,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *msg,
+ k_work_handler_t timer_handler,
+ s32_t timeout, bool need_ack,
+ const struct bt_mesh_send_cb *cb, void *cb_data);
+
+int bt_mesh_client_free_node(sys_slist_t *queue, bt_mesh_client_node_t *node);
+
+#endif /* _BT_MESH_CLIENT_COMMON_H_ */
+
diff --git a/components/bt/ble_mesh/mesh_models/include/common.h b/components/bt/ble_mesh/mesh_models/include/common.h
new file mode 100644
index 0000000000..b3518c00ab
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/include/common.h
@@ -0,0 +1,87 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Model Common APIs.
+ */
+
+#ifndef __BT_MESH_COMMON_H
+#define __BT_MESH_COMMON_H
+
+#include "../../../bluedroid/osi/include/osi/allocator.h"
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+#include "mesh_main.h"
+#include "provisioner_main.h"
+
+struct bt_mesh_common_param {
+ u32_t opcode; /* Message opcode */
+ struct bt_mesh_model *model; /* Pointer to cli structure */
+ struct bt_mesh_msg_ctx ctx; /* Message context */
+ s32_t msg_timeout; /* Time to get response messages */
+ const struct bt_mesh_send_cb *cb; /* User defined callback function */
+ void *cb_data; /* Data as parameter of the cb function */
+};
+
+enum {
+ NODE = 0,
+ PROVISIONER,
+ FAST_PROV,
+};
+
+#define ROLE_NVAL 0xFF
+
+typedef struct bt_mesh_role_param {
+ struct bt_mesh_model *model; /* The client model structure */
+ u8_t role; /* Role of the device - Node/Provisioner */
+} bt_mesh_role_param_t;
+
+/**
+ * @brief This function allocates memory to store outgoing message.
+ *
+ * @param[in] size: Length of memory allocated to store message value
+ *
+ * @return NULL-fail, pointer of a net_buf_simple structure-success
+ */
+struct net_buf_simple *bt_mesh_alloc_buf(u16_t size);
+
+/**
+ * @brief This function releases the memory allocated for the outgoing message.
+ *
+ * @param[in] buf: Pointer to the net_buf_simple structure to be freed
+ *
+ * @return none
+ */
+void bt_mesh_free_buf(struct net_buf_simple *buf);
+
+/**
+ * @brief This function copies node_index for stack internal use.
+ *
+ * @param[in] common: Pointer to the struct bt_mesh_role_param structure
+ *
+ * @return Zero - success, otherwise - fail
+ */
+int bt_mesh_copy_msg_role(bt_mesh_role_param_t *common);
+
+/**
+ * @brief This function gets msg role for stack internal use.
+ *
+ * @param[in] model: Pointer to the model structure
+ * @param[in] srv_send: Indicate if the message is sent by a server model
+ *
+ * @return 0 - Node, 1 - Provisioner
+ */
+u8_t bt_mesh_get_msg_role(struct bt_mesh_model *model, bool srv_send);
+
+#endif /* __BT_MESH_COMMON_H */
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_models/include/generic_client.h b/components/bt/ble_mesh/mesh_models/include/generic_client.h
new file mode 100644
index 0000000000..8e55f567a3
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/include/generic_client.h
@@ -0,0 +1,491 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Generic Client Model APIs.
+ */
+
+#ifndef __BT_MESH_GENERIC_CLI_H
+#define __BT_MESH_GENERIC_CLI_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+#include "common.h"
+#include "bt_mesh_client_common.h"
+
+/* Generic client model common structure */
+typedef bt_mesh_client_common_t bt_mesh_generic_client_t;
+typedef bt_mesh_internal_data_t generic_internal_data_t;
+
+/* Generic OnOff Client Model Context */
+extern const struct bt_mesh_model_op gen_onoff_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_ONOFF_CLI
+ *
+ * Define a new generic onoff client model. Note that this API
+ * needs to be repeated for each element which the application
+ * wants to have a generic onoff client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_onoff_cli.
+ *
+ * @return New generic onoff client model instance.
+ */
+#define BT_MESH_MODEL_GEN_ONOFF_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_ONOFF_CLI, \
+ gen_onoff_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_onoff_cli_t;
+
+struct bt_mesh_gen_onoff_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u8_t present_onoff; /* Present value of Generic OnOff state */
+ u8_t target_onoff; /* Target value of Generic OnOff state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_gen_onoff_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u8_t onoff; /* Target value of Generic OnOff state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+/* Generic Level Client Model Context */
+extern const struct bt_mesh_model_op gen_level_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_LEVEL_CLI
+ *
+ * Define a new generic level client model. Note that this API
+ * needs to be repeated for each element which the application
+ * wants to have a generic level client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_level_cli.
+ *
+ * @return New generic level client model instance.
+ */
+#define BT_MESH_MODEL_GEN_LEVEL_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_LEVEL_CLI, \
+ gen_level_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_level_cli_t;
+
+struct bt_mesh_gen_level_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ s16_t present_level; /* Present value of Generic Level state */
+ s16_t target_level; /* Target value of the Generic Level state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_gen_level_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ s16_t level; /* Target value of Generic Level state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_gen_delta_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ s32_t level; /* Delta change of Generic Level state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_gen_move_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ s16_t delta_level; /* Delta Level step to calculate Move speed for Generic Level state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+/* Generic Default Transition Time Client Model Context */
+extern const struct bt_mesh_model_op gen_def_trans_time_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_DEF_TRANS_TIME_CLI
+ *
+ * Define a new generic default transition time client model. Note
+ * that this API needs to be repeated for each element that the
+ * application wants to have a generic default transition client
+ * model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_def_trans_time_cli.
+ *
+ * @return New generic default transition time client model instance.
+ */
+#define BT_MESH_MODEL_GEN_DEF_TRANS_TIME_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_DEF_TRANS_TIME_CLI, \
+ gen_def_trans_time_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_def_trans_time_cli_t;
+
+struct bt_mesh_gen_def_trans_time_set {
+ u8_t trans_time; /* The value of the Generic Default Transition Time state */
+};
+
+struct bt_mesh_gen_def_trans_time_status {
+ u8_t trans_time; /* The value of the Generic Default Transition Time state */
+};
+
+/* Generic Power OnOff Client Model Context */
+extern const struct bt_mesh_model_op gen_power_onoff_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_POWER_ONOFF_CLI
+ *
+ * Define a new generic power onoff client model. Note that this API
+ * needs to be repeated for each element which the application wants
+ * to have a generic power onoff client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_power_onoff_cli.
+ *
+ * @return New generic power onoff client model instance.
+ */
+#define BT_MESH_MODEL_GEN_POWER_ONOFF_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_POWER_ONOFF_CLI, \
+ gen_power_onoff_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_power_onoff_cli_t;
+
+struct bt_mesh_gen_onpowerup_set {
+ u8_t onpowerup; /* The value of the Generic OnPowerUp state */
+};
+
+struct bt_mesh_gen_onpowerup_status {
+ u8_t onpowerup; /* The value of the Generic OnPowerUp state */
+};
+
+/* Generic Power Level Client Model Context */
+extern const struct bt_mesh_model_op gen_power_level_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_POWER_LEVEL_CLI
+ *
+ * Define a new generic power level client model. Note that this API
+ * needs to be repeated for each element which the application wants
+ * to have a generic power level client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_power_level_cli.
+ *
+ * @return New generic power level client model instance.
+ */
+#define BT_MESH_MODEL_GEN_POWER_LEVEL_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_POWER_LEVEL_CLI, \
+ gen_power_level_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_power_level_cli_t;
+
+struct bt_mesh_gen_power_level_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t present_power; /* Present value of Generic Power Actual state */
+ u16_t target_power; /* Target value of Generic Power Actual state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_gen_power_last_status {
+ u16_t power; /* The value of the Generic Power Last state */
+};
+
+struct bt_mesh_gen_power_default_status {
+ u16_t power; /* The value of the Generic Default Last state */
+};
+
+struct bt_mesh_gen_power_range_status {
+ u8_t status_code; /* Status Code for the requesting message */
+ u16_t range_min; /* Value of Range Min field of Generic Power Range state */
+ u16_t range_max; /* Value of Range Max field of Generic Power Range state */
+};
+
+struct bt_mesh_gen_power_level_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t power; /* Target value of Generic Power Actual state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_gen_power_default_set {
+ u16_t power; /* The value of the Generic Power Default state */
+};
+
+struct bt_mesh_gen_power_range_set {
+ u16_t range_min; /* Value of Range Min field of Generic Power Range state */
+ u16_t range_max; /* Value of Range Max field of Generic Power Range state */
+};
+
+/* Generic Battery Client Model Context */
+extern const struct bt_mesh_model_op gen_battery_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_BATTERY_CLI
+ *
+ * Define a new generic battery client model. Note that this API
+ * needs to be repeated for each element which the application
+ * wants to have a generic battery client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_battery_cli.
+ *
+ * @return New generic battery client model instance.
+ */
+#define BT_MESH_MODEL_GEN_BATTERY_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_BATTERY_CLI, \
+ gen_battery_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_battery_cli_t;
+
+struct bt_mesh_gen_battery_status {
+ u32_t battery_level : 8; /* Value of Generic Battery Level state */
+ u32_t time_to_discharge : 24; /* Value of Generic Battery Time to Discharge state */
+ u32_t time_to_charge : 24; /* Value of Generic Battery Time to Charge state */
+ u32_t flags : 8; /* Value of Generic Battery Flags state */
+};
+
+/* Generic Location Client Model Context */
+extern const struct bt_mesh_model_op gen_location_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_LOCATION_CLI
+ *
+ * Define a new generic location client model. Note that this API
+ * needs to be repeated for each element which the application
+ * wants to have a generic location client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_location_cli.
+ *
+ * @return New generic location client model instance.
+ */
+#define BT_MESH_MODEL_GEN_LOCATION_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_LOCATION_CLI, \
+ gen_location_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_location_cli_t;
+
+struct bt_mesh_gen_loc_global_status {
+ s32_t global_latitude; /* Global Coordinates (Latitude) */
+ s32_t global_longitude; /* Global Coordinates (Longitude) */
+ s16_t global_altitude; /* Global Altitude */
+};
+
+struct bt_mesh_gen_loc_local_status {
+ s16_t local_north; /* Local Coordinates (North) */
+ s16_t local_east; /* Local Coordinates (East) */
+ s16_t local_altitude; /* Local Altitude */
+ u8_t floor_number; /* Floor Number */
+ u16_t uncertainty; /* Uncertainty */
+};
+
+struct bt_mesh_gen_loc_global_set {
+ s32_t global_latitude; /* Global Coordinates (Latitude) */
+ s32_t global_longitude; /* Global Coordinates (Longitude) */
+ s16_t global_altitude; /* Global Altitude */
+};
+
+struct bt_mesh_gen_loc_local_set {
+ s16_t local_north; /* Local Coordinates (North) */
+ s16_t local_east; /* Local Coordinates (East) */
+ s16_t local_altitude; /* Local Altitude */
+ u8_t floor_number; /* Floor Number */
+ u16_t uncertainty; /* Uncertainty */
+};
+
+/* Generic Property Client Model Context */
+extern const struct bt_mesh_model_op gen_property_cli_op[];
+
+/** @def BT_MESH_MODEL_GEN_LOCATION_CLI
+ *
+ * Define a new generic location client model. Note that this API
+ * needs to be repeated for each element which the application
+ * wants to have a generic location client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_gen_location_cli.
+ *
+ * @return New generic location client model instance.
+ */
+#define BT_MESH_MODEL_GEN_PROPERTY_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_GEN_PROP_CLI, \
+ gen_property_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_gen_property_cli_t;
+
+struct bt_mesh_gen_user_properties_status {
+ struct net_buf_simple *user_property_ids; /* Buffer contains a sequence of N User Property IDs */
+};
+
+struct bt_mesh_gen_user_property_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t user_property_id; /* Property ID identifying a Generic User Property */
+ u8_t user_access; /* Enumeration indicating user access (optional) */
+ struct net_buf_simple *user_property_value; /* Raw value for the User Property (C.1) */
+};
+
+struct bt_mesh_gen_admin_properties_status {
+ struct net_buf_simple *admin_property_ids; /* Buffer contains a sequence of N Admin Property IDs */
+};
+
+struct bt_mesh_gen_admin_property_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t admin_property_id; /* Property ID identifying a Generic Admin Property */
+ u8_t admin_user_access; /* Enumeration indicating user access (optional) */
+ struct net_buf_simple *admin_property_value; /* Raw value for the Admin Property (C.1) */
+};
+
+struct bt_mesh_gen_manu_properties_status {
+ struct net_buf_simple *manu_property_ids; /* Buffer contains a sequence of N Manufacturer Property IDs */
+};
+
+struct bt_mesh_gen_manu_property_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t manu_property_id; /* Property ID identifying a Generic Manufacturer Property */
+ u8_t manu_user_access; /* Enumeration indicating user access (optional) */
+ struct net_buf_simple *manu_property_value; /* Raw value for the Manufacturer Property (C.1) */
+};
+
+struct bt_mesh_gen_client_properties_status {
+ struct net_buf_simple *client_property_ids; /* Buffer contains a sequence of N Client Property IDs */
+};
+
+struct bt_mesh_gen_user_property_get {
+ u16_t user_property_id; /* Property ID identifying a Generic User Property */
+};
+
+struct bt_mesh_gen_user_property_set {
+ u16_t user_property_id; /* Property ID identifying a Generic User Property */
+ struct net_buf_simple *user_property_value; /* Raw value for the User Property */
+};
+
+struct bt_mesh_gen_admin_property_get {
+ u16_t admin_property_id; /* Property ID identifying a Generic Admin Property */
+};
+
+struct bt_mesh_gen_admin_property_set {
+ u16_t admin_property_id; /* Property ID identifying a Generic Admin Property */
+ u8_t admin_user_access; /* Enumeration indicating user access */
+ struct net_buf_simple *admin_property_value; /* Raw value for the Admin Property */
+};
+
+struct bt_mesh_gen_manu_property_get {
+ u16_t manu_property_id; /* Property ID identifying a Generic Manufacturer Property */
+};
+
+struct bt_mesh_gen_manu_property_set {
+ u16_t manu_property_id; /* Property ID identifying a Generic Manufacturer Property */
+ u8_t manu_user_access; /* Enumeration indicating user access */
+};
+
+struct bt_mesh_gen_client_properties_get {
+ u16_t client_property_id; /* A starting Client Property ID present within an element */
+};
+
+/**
+ * @brief This function is called to initialize generic onoff client model user_data.
+ *
+ * @param[in] model: Pointer to generic onoff client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_onoff_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize generic level client model user_data.
+ *
+ * @param[in] model: Pointer to generic level client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_level_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize generic default transition time
+ * client model user_data.
+ *
+ * @param[in] model: Pointer to generic default transition time client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_def_trans_time_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize generic power onoff client model user_data.
+ *
+ * @param[in] model: Pointer to generic power onoff client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_pwr_onoff_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize generic power level client model user_data.
+ *
+ * @param[in] model: Pointer to generic power level client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_pwr_level_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize generic battery client model user_data.
+ *
+ * @param[in] model: Pointer to generic battery client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_battery_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize generic location client model user_data.
+ *
+ * @param[in] model: Pointer to generic location client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_location_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize generic property client model user_data.
+ *
+ * @param[in] model: Pointer to generic property client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_gen_property_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to get generic states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] get: Pointer of generic get message value
+ * @param[out] status: Pointer of generic status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_generic_client_get_state(struct bt_mesh_common_param *common, void *get, void *status);
+
+/**
+ * @brief This function is called to set generic states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] set: Pointer of generic set message value
+ * @param[out] status: Pointer of generic status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_generic_client_set_state(struct bt_mesh_common_param *common, void *set, void *status);
+
+#endif /* __BT_MESH_GENERIC_CLI_H */
diff --git a/components/bt/ble_mesh/mesh_models/include/light_client.h b/components/bt/ble_mesh/mesh_models/include/light_client.h
new file mode 100644
index 0000000000..1accc1e84f
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/include/light_client.h
@@ -0,0 +1,492 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Lighting Client Model APIs.
+ */
+
+#ifndef __BT_MESH_LIGHTING_CLI_H
+#define __BT_MESH_LIGHTING_CLI_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+#include "common.h"
+#include "bt_mesh_client_common.h"
+
+/* Light client model common structure */
+typedef bt_mesh_client_common_t bt_mesh_light_client_t;
+typedef bt_mesh_internal_data_t light_internal_data_t;
+
+/* Light Lightness Client Model Context */
+extern const struct bt_mesh_model_op light_lightness_cli_op[];
+
+/** @def BT_MESH_MODEL_LIGHT_LIGHTNESS_CLI
+ *
+ * Define a new light lightness client model. Note that this API
+ * needs to be repeated for each element which the application
+ * wants to have a light lightness client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_light_lightness_cli.
+ *
+ * @return New light lightness client model instance.
+ */
+#define BT_MESH_MODEL_LIGHT_LIGHTNESS_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_LIGHT_LIGHTNESS_CLI, \
+ light_lightness_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_light_lightness_cli_t;
+
+struct bt_mesh_light_lightness_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t present_lightness; /* Present value of light lightness actual state */
+ u16_t target_lightness; /* Target value of light lightness actual state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_light_lightness_linear_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t present_lightness; /* Present value of light lightness linear state */
+ u16_t target_lightness; /* Target value of light lightness linear state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_light_lightness_last_status {
+ u16_t lightness; /* The value of the Light Lightness Last state */
+};
+
+struct bt_mesh_light_lightness_default_status {
+ u16_t lightness; /* The value of the Light Lightness default state */
+};
+
+struct bt_mesh_light_lightness_range_status {
+ u8_t status_code; /* Status Code for the requesting message */
+ u16_t range_min; /* Value of range min field of light lightness range state */
+ u16_t range_max; /* Value of range max field of light lightness range state */
+};
+
+struct bt_mesh_light_lightness_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t lightness; /* Target value of light lightness actual state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_lightness_linear_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t lightness; /* Target value of light lightness linear state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_lightness_default_set {
+ u16_t lightness; /* The value of the Light Lightness Default state */
+};
+
+struct bt_mesh_light_lightness_range_set {
+ u16_t range_min; /* Value of range min field of light lightness range state */
+ u16_t range_max; /* Value of range max field of light lightness range state */
+};
+
+/* Light CTL Client Model Context */
+extern const struct bt_mesh_model_op light_ctl_cli_op[];
+
+/** @def BT_MESH_MODEL_LIGHT_CTL_CLI
+ *
+ * Define a new light CTL client model. Note that this API needs
+ * to be repeated for each element which the application wants to
+ * have a light CTL client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_light_ctl_cli.
+ *
+ * @return New light CTL client model instance.
+ */
+#define BT_MESH_MODEL_LIGHT_CTL_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_LIGHT_CTL_CLI, \
+ light_ctl_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_light_ctl_cli_t;
+
+struct bt_mesh_light_ctl_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t present_ctl_lightness; /* Present value of light ctl lightness state */
+ u16_t present_ctl_temperature; /* Present value of light ctl temperature state */
+ u16_t target_ctl_lightness; /* Target value of light ctl lightness state (optional) */
+ u16_t target_ctl_temperature; /* Target value of light ctl temperature state (C.1) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_light_ctl_temperature_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t present_ctl_temperature; /* Present value of light ctl temperature state */
+ u16_t present_ctl_delta_uv; /* Present value of light ctl delta UV state */
+ u16_t target_ctl_temperature; /* Target value of light ctl temperature state (optional) */
+ u16_t target_ctl_delta_uv; /* Target value of light ctl delta UV state (C.1) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_light_ctl_temperature_range_status {
+ u8_t status_code; /* Status code for the requesting message */
+ u16_t range_min; /* Value of temperature range min field of light ctl temperature range state */
+ u16_t range_max; /* Value of temperature range max field of light ctl temperature range state */
+};
+
+struct bt_mesh_light_ctl_default_status {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t temperature; /* Value of light temperature default state */
+ s16_t delta_uv; /* Value of light delta UV default state */
+};
+
+struct bt_mesh_light_ctl_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t ctl_lightness; /* Target value of light ctl lightness state */
+ u16_t ctl_temperature; /* Target value of light ctl temperature state */
+ s16_t ctl_delta_uv; /* Target value of light ctl delta UV state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_ctl_temperature_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t ctl_temperature; /* Target value of light ctl temperature state */
+ s16_t ctl_delta_uv; /* Target value of light ctl delta UV state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_ctl_temperature_range_set {
+ u16_t range_min; /* Value of temperature range min field of light ctl temperature range state */
+ u16_t range_max; /* Value of temperature range max field of light ctl temperature range state */
+};
+
+struct bt_mesh_light_ctl_default_set {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t temperature; /* Value of light temperature default state */
+ s16_t delta_uv; /* Value of light delta UV default state */
+};
+
+/* Light HSL Client Model Context */
+extern const struct bt_mesh_model_op light_hsl_cli_op[];
+
+/** @def BT_MESH_MODEL_LIGHT_HSL_CLI
+ *
+ * Define a new light HSL client model. Note that this API needs
+ * to be repeated for each element which the application wants to
+ * have a light HSL client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_light_hsl_cli.
+ *
+ * @return New light HSL client model instance.
+ */
+#define BT_MESH_MODEL_LIGHT_HSL_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_LIGHT_HSL_CLI, \
+ light_hsl_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_light_hsl_cli_t;
+
+struct bt_mesh_light_hsl_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t hsl_lightness; /* Present value of light hsl lightness state */
+ u16_t hsl_hue; /* Present value of light hsl hue state */
+ u16_t hsl_saturation; /* Present value of light hsl saturation state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+};
+
+struct bt_mesh_light_hsl_target_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t hsl_lightness_target; /* Target value of light hsl lightness state */
+ u16_t hsl_hue_target; /* Target value of light hsl hue state */
+ u16_t hsl_saturation_target; /* Target value of light hsl saturation state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+};
+
+struct bt_mesh_light_hsl_hue_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t present_hue; /* Present value of light hsl hue state */
+ u16_t target_hue; /* Target value of light hsl hue state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_light_hsl_saturation_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t present_saturation; /* Present value of light hsl saturation state */
+ u16_t target_saturation; /* Target value of light hsl saturation state (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_light_hsl_default_status {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t hue; /* Value of light hue default state */
+ u16_t saturation; /* Value of light saturation default state */
+};
+
+struct bt_mesh_light_hsl_range_status {
+ u8_t status_code; /* Status code for the requesting message */
+ u16_t hue_range_min; /* Value of hue range min field of light hsl hue range state */
+ u16_t hue_range_max; /* Value of hue range max field of light hsl hue range state */
+ u16_t saturation_range_min; /* Value of saturation range min field of light hsl saturation range state */
+ u16_t saturation_range_max; /* Value of saturation range max field of light hsl saturation range state */
+};
+
+struct bt_mesh_light_hsl_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t hsl_lightness; /* Target value of light hsl lightness state */
+ u16_t hsl_hue; /* Target value of light hsl hue state */
+ u16_t hsl_saturation; /* Target value of light hsl saturation state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_hsl_hue_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t hue; /* Target value of light hsl hue state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_hsl_saturation_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t saturation; /* Target value of light hsl hue state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_hsl_default_set {
+ u16_t lightness; /* Value of light lightness default state */
+ u16_t hue; /* Value of light hue default state */
+ u16_t saturation; /* Value of light saturation default state */
+};
+
+struct bt_mesh_light_hsl_range_set {
+ u16_t hue_range_min; /* Value of hue range min field of light hsl hue range state */
+ u16_t hue_range_max; /* Value of hue range max field of light hsl hue range state */
+ u16_t saturation_range_min; /* Value of saturation range min field of light hsl saturation range state */
+ u16_t saturation_range_max; /* Value of saturation range max field of light hsl saturation range state */
+};
+
+/* Light xyL Client Model Context */
+extern const struct bt_mesh_model_op light_xyl_cli_op[];
+
+/** @def BT_MESH_MODEL_LIGHT_XYL_CLI
+ *
+ * Define a new light xyL client model. Note that this API needs
+ * to be repeated for each element which the application wants
+ * to have a light xyL client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_light_xyl_cli.
+ *
+ * @return New light xyL client model instance.
+ */
+#define BT_MESH_MODEL_LIGHT_XYL_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_LIGHT_XYL_CLI, \
+ light_xyl_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_light_xyl_cli_t;
+
+struct bt_mesh_light_xyl_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t xyl_lightness; /* The present value of the Light xyL Lightness state */
+ u16_t xyl_x; /* The present value of the Light xyL x state */
+ u16_t xyl_y; /* The present value of the Light xyL y state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+};
+
+struct bt_mesh_light_xyl_target_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t target_xyl_lightness; /* The target value of the Light xyL Lightness state */
+ u16_t target_xyl_x; /* The target value of the Light xyL x state */
+ u16_t target_xyl_y; /* The target value of the Light xyL y state */
+ u8_t remain_time; /* Time to complete state transition (optional) */
+};
+
+struct bt_mesh_light_xyl_default_status {
+ u16_t lightness; /* The value of the Light Lightness Default state */
+ u16_t xyl_x; /* The value of the Light xyL x Default state */
+ u16_t xyl_y; /* The value of the Light xyL y Default state */
+};
+
+struct bt_mesh_light_xyl_range_status {
+ u8_t status_code; /* Status Code for the requesting message */
+ u16_t xyl_x_range_min; /* The value of the xyL x Range Min field of the Light xyL x Range state */
+ u16_t xyl_x_range_max; /* The value of the xyL x Range Max field of the Light xyL x Range state */
+ u16_t xyl_y_range_min; /* The value of the xyL y Range Min field of the Light xyL y Range state */
+ u16_t xyl_y_range_max; /* The value of the xyL y Range Max field of the Light xyL y Range state */
+};
+
+struct bt_mesh_light_xyl_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t xyl_lightness; /* The target value of the Light xyL Lightness state */
+ u16_t xyl_x; /* The target value of the Light xyL x state */
+ u16_t xyl_y; /* The target value of the Light xyL y state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_xyl_default_set {
+ u16_t lightness; /* The value of the Light Lightness Default state */
+ u16_t xyl_x; /* The value of the Light xyL x Default state */
+ u16_t xyl_y; /* The value of the Light xyL y Default state */
+};
+
+struct bt_mesh_light_xyl_range_set {
+ u16_t xyl_x_range_min; /* The value of the xyL x Range Min field of the Light xyL x Range state */
+ u16_t xyl_x_range_max; /* The value of the xyL x Range Max field of the Light xyL x Range state */
+ u16_t xyl_y_range_min; /* The value of the xyL y Range Min field of the Light xyL y Range state */
+ u16_t xyl_y_range_max; /* The value of the xyL y Range Max field of the Light xyL y Range state */
+};
+
+/* Light LC Client Model Context */
+extern const struct bt_mesh_model_op light_lc_cli_op[];
+
+/** @def BT_MESH_MODEL_LIGHT_LC_CLI
+ *
+ * Define a new light lc client model. Note that this API needs
+ * to be repeated for each element which the application wants
+ * to have a light lc client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_light_lc_cli.
+ *
+ * @return New light lc client model instance.
+ */
+#define BT_MESH_MODEL_LIGHT_LC_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_LIGHT_LC_CLI, \
+ light_lc_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_light_lc_cli_t;
+
+struct bt_mesh_light_lc_mode_status {
+ u8_t mode; /* The present value of the Light LC Mode state */
+};
+
+struct bt_mesh_light_lc_om_status {
+ u8_t mode; /* The present value of the Light LC Occupancy Mode state */
+};
+
+struct bt_mesh_light_lc_light_onoff_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u8_t present_light_onoff; /* The present value of the Light LC Light OnOff state */
+ u8_t target_light_onoff; /* The target value of the Light LC Light OnOff state (Optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_light_lc_property_status {
+ u16_t light_lc_property_id; /* Property ID identifying a Light LC Property */
+ struct net_buf_simple *light_lc_property_value; /* Raw value for the Light LC Property */
+};
+
+struct bt_mesh_light_lc_mode_set {
+ u8_t mode; /* The target value of the Light LC Mode state */
+};
+
+struct bt_mesh_light_lc_om_set {
+ u8_t mode; /* The target value of the Light LC Occupancy Mode state */
+};
+
+struct bt_mesh_light_lc_light_onoff_set {
+ bool op_en; /* Indicate whether optional parameters included */
+ u8_t light_onoff; /* The target value of the Light LC Light OnOff state */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_light_lc_property_get {
+ u16_t light_lc_property_id; /* Property ID identifying a Light LC Property */
+};
+
+struct bt_mesh_light_lc_property_set {
+ u16_t light_lc_property_id; /* Property ID identifying a Light LC Property */
+ struct net_buf_simple *light_lc_property_value; /* Raw value for the Light LC Property */
+};
+
+/**
+ * @brief This function is called to initialize light lightness client model user_data.
+ *
+ * @param[in] model: Pointer to light lightness client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_light_lightness_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize light ctl client model user_data.
+ *
+ * @param[in] model: Pointer to light ctl client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_light_ctl_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize light hsl client model user_data.
+ *
+ * @param[in] model: Pointer to light hsl client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_light_hsl_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize light xyl client model user_data.
+ *
+ * @param[in] model: Pointer to light xyl client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_light_xyl_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize light lc client model user_data.
+ *
+ * @param[in] model: Pointer to light lc client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_light_lc_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to get light states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] get: Pointer of light get message value
+ * @param[out] status: Pointer of light status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_light_client_get_state(struct bt_mesh_common_param *common, void *get, void *status);
+
+/**
+ * @brief This function is called to set light states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] set: Pointer of light set message value
+ * @param[out] status: Pointer of light status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_light_client_set_state(struct bt_mesh_common_param *common, void *set, void *status);
+
+#endif /* __BT_MESH_LIGHTING_CLI_H */
diff --git a/components/bt/ble_mesh/mesh_models/include/model_op.h b/components/bt/ble_mesh/mesh_models/include/model_op.h
new file mode 100644
index 0000000000..7eb9650b88
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/include/model_op.h
@@ -0,0 +1,276 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef __BT_MESH_MODEL_OP_H_
+#define __BT_MESH_MODEL_OP_H_
+
+#include "mesh_main.h"
+
+/* Generic OnOff Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_ONOFF_GET BT_MESH_MODEL_OP_2(0x82, 0x01)
+#define BT_MESH_MODEL_OP_GEN_ONOFF_SET BT_MESH_MODEL_OP_2(0x82, 0x02)
+#define BT_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x03)
+#define BT_MESH_MODEL_OP_GEN_ONOFF_STATUS BT_MESH_MODEL_OP_2(0x82, 0x04)
+
+/* Generic Level Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_LEVEL_GET BT_MESH_MODEL_OP_2(0x82, 0x05)
+#define BT_MESH_MODEL_OP_GEN_LEVEL_SET BT_MESH_MODEL_OP_2(0x82, 0x06)
+#define BT_MESH_MODEL_OP_GEN_LEVEL_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x07)
+#define BT_MESH_MODEL_OP_GEN_LEVEL_STATUS BT_MESH_MODEL_OP_2(0x82, 0x08)
+#define BT_MESH_MODEL_OP_GEN_DELTA_SET BT_MESH_MODEL_OP_2(0x82, 0x09)
+#define BT_MESH_MODEL_OP_GEN_DELTA_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x0A)
+#define BT_MESH_MODEL_OP_GEN_MOVE_SET BT_MESH_MODEL_OP_2(0x82, 0x0B)
+#define BT_MESH_MODEL_OP_GEN_MOVE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x0C)
+
+/* Generic Default Transition Time Message Opcode*/
+#define BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_GET BT_MESH_MODEL_OP_2(0x82, 0x0D)
+#define BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET BT_MESH_MODEL_OP_2(0x82, 0x0E)
+#define BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x0F)
+#define BT_MESH_MODEL_OP_GEN_DEF_TRANS_TIME_STATUS BT_MESH_MODEL_OP_2(0x82, 0x10)
+
+/* Generic Power OnOff Message Opcode*/
+#define BT_MESH_MODEL_OP_GEN_ONPOWERUP_GET BT_MESH_MODEL_OP_2(0x82, 0x11)
+#define BT_MESH_MODEL_OP_GEN_ONPOWERUP_STATUS BT_MESH_MODEL_OP_2(0x82, 0x12)
+
+/* Generic Power OnOff Setup Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET BT_MESH_MODEL_OP_2(0x82, 0x13)
+#define BT_MESH_MODEL_OP_GEN_ONPOWERUP_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x14)
+
+/* Generic Power Level Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_POWER_LEVEL_GET BT_MESH_MODEL_OP_2(0x82, 0x15)
+#define BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET BT_MESH_MODEL_OP_2(0x82, 0x16)
+#define BT_MESH_MODEL_OP_GEN_POWER_LEVEL_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x17)
+#define BT_MESH_MODEL_OP_GEN_POWER_LEVEL_STATUS BT_MESH_MODEL_OP_2(0x82, 0x18)
+#define BT_MESH_MODEL_OP_GEN_POWER_LAST_GET BT_MESH_MODEL_OP_2(0x82, 0x19)
+#define BT_MESH_MODEL_OP_GEN_POWER_LAST_STATUS BT_MESH_MODEL_OP_2(0x82, 0x1A)
+#define BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_GET BT_MESH_MODEL_OP_2(0x82, 0x1B)
+#define BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_STATUS BT_MESH_MODEL_OP_2(0x82, 0x1C)
+#define BT_MESH_MODEL_OP_GEN_POWER_RANGE_GET BT_MESH_MODEL_OP_2(0x82, 0x1D)
+#define BT_MESH_MODEL_OP_GEN_POWER_RANGE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x1E)
+
+/* Generic Power Level Setup Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET BT_MESH_MODEL_OP_2(0x82, 0x1F)
+#define BT_MESH_MODEL_OP_GEN_POWER_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x20)
+#define BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET BT_MESH_MODEL_OP_2(0x82, 0x21)
+#define BT_MESH_MODEL_OP_GEN_POWER_RANGE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x22)
+
+/* Generic Battery Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_BATTERY_GET BT_MESH_MODEL_OP_2(0x82, 0x23)
+#define BT_MESH_MODEL_OP_GEN_BATTERY_STATUS BT_MESH_MODEL_OP_2(0x82, 0x24)
+
+/* Generic Location Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_GET BT_MESH_MODEL_OP_2(0x82, 0x25)
+#define BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_STATUS BT_MESH_MODEL_OP_1(0x40)
+#define BT_MESH_MODEL_OP_GEN_LOC_LOCAL_GET BT_MESH_MODEL_OP_2(0x82, 0x26)
+#define BT_MESH_MODEL_OP_GEN_LOC_LOCAL_STATUS BT_MESH_MODEL_OP_2(0x82, 0x27)
+
+/* Generic Location Setup Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET BT_MESH_MODEL_OP_1(0x41)
+#define BT_MESH_MODEL_OP_GEN_LOC_GLOBAL_SET_UNACK BT_MESH_MODEL_OP_1(0x42)
+#define BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET BT_MESH_MODEL_OP_2(0x82, 0x28)
+#define BT_MESH_MODEL_OP_GEN_LOC_LOCAL_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x29)
+
+/* Generic Manufacturer Property Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_GET BT_MESH_MODEL_OP_2(0x82, 0x2A)
+#define BT_MESH_MODEL_OP_GEN_MANU_PROPERTIES_STATUS BT_MESH_MODEL_OP_1(0x43)
+#define BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_GET BT_MESH_MODEL_OP_2(0x82, 0x2B)
+#define BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET BT_MESH_MODEL_OP_1(0x44)
+#define BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_1(0x45)
+#define BT_MESH_MODEL_OP_GEN_MANU_PROPERTY_STATUS BT_MESH_MODEL_OP_1(0x46)
+
+/* Generic Admin Property Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_GET BT_MESH_MODEL_OP_2(0x82, 0x2C)
+#define BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTIES_STATUS BT_MESH_MODEL_OP_1(0x47)
+#define BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_GET BT_MESH_MODEL_OP_2(0x82, 0x2D)
+#define BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET BT_MESH_MODEL_OP_1(0x48)
+#define BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_1(0x49)
+#define BT_MESH_MODEL_OP_GEN_ADMIN_PROPERTY_STATUS BT_MESH_MODEL_OP_1(0x4A)
+
+/* Generic User Property Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_GET BT_MESH_MODEL_OP_2(0x82, 0x2E)
+#define BT_MESH_MODEL_OP_GEN_USER_PROPERTIES_STATUS BT_MESH_MODEL_OP_1(0x4B)
+#define BT_MESH_MODEL_OP_GEN_USER_PROPERTY_GET BT_MESH_MODEL_OP_2(0x82, 0x2F)
+#define BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET BT_MESH_MODEL_OP_1(0x4C)
+#define BT_MESH_MODEL_OP_GEN_USER_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_1(0x4D)
+#define BT_MESH_MODEL_OP_GEN_USER_PROPERTY_STATUS BT_MESH_MODEL_OP_1(0x4E)
+
+/* Generic Client Property Message Opcode */
+#define BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_GET BT_MESH_MODEL_OP_1(0x4F)
+#define BT_MESH_MODEL_OP_GEN_CLIENT_PROPERTIES_STATUS BT_MESH_MODEL_OP_1(0x50)
+
+/* Sensor Message Opcode */
+#define BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET BT_MESH_MODEL_OP_2(0x82, 0x30)
+#define BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS BT_MESH_MODEL_OP_1(0x51)
+#define BT_MESH_MODEL_OP_SENSOR_GET BT_MESH_MODEL_OP_2(0x82, 0x31)
+#define BT_MESH_MODEL_OP_SENSOR_STATUS BT_MESH_MODEL_OP_1(0x52)
+#define BT_MESH_MODEL_OP_SENSOR_COLUMN_GET BT_MESH_MODEL_OP_2(0x82, 0x32)
+#define BT_MESH_MODEL_OP_SENSOR_COLUMN_STATUS BT_MESH_MODEL_OP_1(0x53)
+#define BT_MESH_MODEL_OP_SENSOR_SERIES_GET BT_MESH_MODEL_OP_2(0x82, 0x33)
+#define BT_MESH_MODEL_OP_SENSOR_SERIES_STATUS BT_MESH_MODEL_OP_1(0x54)
+
+/* Sensor Setup Message Opcode */
+#define BT_MESH_MODEL_OP_SENSOR_CADENCE_GET BT_MESH_MODEL_OP_2(0x82, 0x34)
+#define BT_MESH_MODEL_OP_SENSOR_CADENCE_SET BT_MESH_MODEL_OP_1(0x55)
+#define BT_MESH_MODEL_OP_SENSOR_CADENCE_SET_UNACK BT_MESH_MODEL_OP_1(0x56)
+#define BT_MESH_MODEL_OP_SENSOR_CADENCE_STATUS BT_MESH_MODEL_OP_1(0x57)
+#define BT_MESH_MODEL_OP_SENSOR_SETTINGS_GET BT_MESH_MODEL_OP_2(0x82, 0x35)
+#define BT_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS BT_MESH_MODEL_OP_1(0x58)
+#define BT_MESH_MODEL_OP_SENSOR_SETTING_GET BT_MESH_MODEL_OP_2(0x82, 0x36)
+#define BT_MESH_MODEL_OP_SENSOR_SETTING_SET BT_MESH_MODEL_OP_1(0x59)
+#define BT_MESH_MODEL_OP_SENSOR_SETTING_SET_UNACK BT_MESH_MODEL_OP_1(0x5A)
+#define BT_MESH_MODEL_OP_SENSOR_SETTING_STATUS BT_MESH_MODEL_OP_1(0x5B)
+
+/* Time Message Opcode */
+#define BT_MESH_MODEL_OP_TIME_GET BT_MESH_MODEL_OP_2(0x82, 0x37)
+#define BT_MESH_MODEL_OP_TIME_SET BT_MESH_MODEL_OP_1(0x5C)
+#define BT_MESH_MODEL_OP_TIME_STATUS BT_MESH_MODEL_OP_1(0x5D)
+#define BT_MESH_MODEL_OP_TIME_ROLE_GET BT_MESH_MODEL_OP_2(0x82, 0x38)
+#define BT_MESH_MODEL_OP_TIME_ROLE_SET BT_MESH_MODEL_OP_2(0x82, 0x39)
+#define BT_MESH_MODEL_OP_TIME_ROLE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x3A)
+#define BT_MESH_MODEL_OP_TIME_ZONE_GET BT_MESH_MODEL_OP_2(0x82, 0x3B)
+#define BT_MESH_MODEL_OP_TIME_ZONE_SET BT_MESH_MODEL_OP_2(0x82, 0x3C)
+#define BT_MESH_MODEL_OP_TIME_ZONE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x3D)
+#define BT_MESH_MODEL_OP_TAI_UTC_DELTA_GET BT_MESH_MODEL_OP_2(0x82, 0x3E)
+#define BT_MESH_MODEL_OP_TAI_UTC_DELTA_SET BT_MESH_MODEL_OP_2(0x82, 0x3F)
+#define BT_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS BT_MESH_MODEL_OP_2(0x82, 0x40)
+
+/* Scene Message Opcode */
+#define BT_MESH_MODEL_OP_SCENE_GET BT_MESH_MODEL_OP_2(0x82, 0x41)
+#define BT_MESH_MODEL_OP_SCENE_RECALL BT_MESH_MODEL_OP_2(0x82, 0x42)
+#define BT_MESH_MODEL_OP_SCENE_RECALL_UNACK BT_MESH_MODEL_OP_2(0x82, 0x43)
+#define BT_MESH_MODEL_OP_SCENE_STATUS BT_MESH_MODEL_OP_1(0x5E)
+#define BT_MESH_MODEL_OP_SCENE_REGISTER_GET BT_MESH_MODEL_OP_2(0x82, 0x44)
+#define BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS BT_MESH_MODEL_OP_2(0x82, 0x45)
+
+/* Scene Setup Message Opcode */
+#define BT_MESH_MODEL_OP_SCENE_STORE BT_MESH_MODEL_OP_2(0x82, 0x46)
+#define BT_MESH_MODEL_OP_SCENE_STORE_UNACK BT_MESH_MODEL_OP_2(0x82, 0x47)
+#define BT_MESH_MODEL_OP_SCENE_DELETE BT_MESH_MODEL_OP_2(0x82, 0x9E)
+#define BT_MESH_MODEL_OP_SCENE_DELETE_UNACK BT_MESH_MODEL_OP_2(0x82, 0x9F)
+
+/* Scheduler Message Opcode */
+#define BT_MESH_MODEL_OP_SCHEDULER_ACT_GET BT_MESH_MODEL_OP_2(0x82, 0x48)
+#define BT_MESH_MODEL_OP_SCHEDULER_ACT_STATUS BT_MESH_MODEL_OP_1(0x5F)
+#define BT_MESH_MODEL_OP_SCHEDULER_GET BT_MESH_MODEL_OP_2(0x82, 0x49)
+#define BT_MESH_MODEL_OP_SCHEDULER_STATUS BT_MESH_MODEL_OP_2(0x82, 0x4A)
+
+/* Scheduler Setup Message Opcode */
+#define BT_MESH_MODEL_OP_SCHEDULER_ACT_SET BT_MESH_MODEL_OP_1(0x60)
+#define BT_MESH_MODEL_OP_SCHEDULER_ACT_SET_UNACK BT_MESH_MODEL_OP_1(0x61)
+
+/* Light Lightness Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET BT_MESH_MODEL_OP_2(0x82, 0x4B)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET BT_MESH_MODEL_OP_2(0x82, 0x4C)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x4D)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS BT_MESH_MODEL_OP_2(0x82, 0x4E)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET BT_MESH_MODEL_OP_2(0x82, 0x4F)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET BT_MESH_MODEL_OP_2(0x82, 0x50)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x51)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS BT_MESH_MODEL_OP_2(0x82, 0x52)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET BT_MESH_MODEL_OP_2(0x82, 0x53)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS BT_MESH_MODEL_OP_2(0x82, 0x54)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET BT_MESH_MODEL_OP_2(0x82, 0x55)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS BT_MESH_MODEL_OP_2(0x82, 0x56)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET BT_MESH_MODEL_OP_2(0x82, 0x57)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x58)
+
+/* Light Lightness Setup Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET BT_MESH_MODEL_OP_2(0x82, 0x59)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x5A)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET BT_MESH_MODEL_OP_2(0x82, 0x5B)
+#define BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x5C)
+
+/* Light CTL Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_CTL_GET BT_MESH_MODEL_OP_2(0x82, 0x5D)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_SET BT_MESH_MODEL_OP_2(0x82, 0x5E)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x5F)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_STATUS BT_MESH_MODEL_OP_2(0x82, 0x60)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET BT_MESH_MODEL_OP_2(0x82, 0x61)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET BT_MESH_MODEL_OP_2(0x82, 0x62)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x63)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET BT_MESH_MODEL_OP_2(0x82, 0x64)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x65)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x66)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET BT_MESH_MODEL_OP_2(0x82, 0x67)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS BT_MESH_MODEL_OP_2(0x82, 0x68)
+
+/* Light CTL Setup Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET BT_MESH_MODEL_OP_2(0x82, 0x69)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x6A)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET BT_MESH_MODEL_OP_2(0x82, 0x6B)
+#define BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x6C)
+
+/* Light HSL Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_HSL_GET BT_MESH_MODEL_OP_2(0x82, 0x6D)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_HUE_GET BT_MESH_MODEL_OP_2(0x82, 0x6E)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET BT_MESH_MODEL_OP_2(0x82, 0x6F)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x70)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x71)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET BT_MESH_MODEL_OP_2(0x82, 0x72)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET BT_MESH_MODEL_OP_2(0x82, 0x73)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x74)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS BT_MESH_MODEL_OP_2(0x82, 0x75)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_SET BT_MESH_MODEL_OP_2(0x82, 0x76)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x77)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_STATUS BT_MESH_MODEL_OP_2(0x82, 0x78)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET BT_MESH_MODEL_OP_2(0x82, 0x79)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS BT_MESH_MODEL_OP_2(0x82, 0x7A)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET BT_MESH_MODEL_OP_2(0x82, 0x7B)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS BT_MESH_MODEL_OP_2(0x82, 0x7C)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET BT_MESH_MODEL_OP_2(0x82, 0x7D)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x7E)
+
+/* Light HSL Setup Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET BT_MESH_MODEL_OP_2(0x82, 0x7F)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x80)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET BT_MESH_MODEL_OP_2(0x82, 0x81)
+#define BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x82) /* Model spec is wrong */
+
+/* Light xyL Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_XYL_GET BT_MESH_MODEL_OP_2(0x82, 0x83)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_SET BT_MESH_MODEL_OP_2(0x82, 0x84)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x85)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_STATUS BT_MESH_MODEL_OP_2(0x82, 0x86)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET BT_MESH_MODEL_OP_2(0x82, 0x87)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS BT_MESH_MODEL_OP_2(0x82, 0x88)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET BT_MESH_MODEL_OP_2(0x82, 0x89)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS BT_MESH_MODEL_OP_2(0x82, 0x8A)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET BT_MESH_MODEL_OP_2(0x82, 0x8B)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x8C)
+
+/* Light xyL Setup Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET BT_MESH_MODEL_OP_2(0x82, 0x8D)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x8E)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET BT_MESH_MODEL_OP_2(0x82, 0x8F)
+#define BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x90)
+
+/* Light Control Message Opcode */
+#define BT_MESH_MODEL_OP_LIGHT_LC_MODE_GET BT_MESH_MODEL_OP_2(0x82, 0x91)
+#define BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET BT_MESH_MODEL_OP_2(0x82, 0x92)
+#define BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x93)
+#define BT_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS BT_MESH_MODEL_OP_2(0x82, 0x94)
+#define BT_MESH_MODEL_OP_LIGHT_LC_OM_GET BT_MESH_MODEL_OP_2(0x82, 0x95)
+#define BT_MESH_MODEL_OP_LIGHT_LC_OM_SET BT_MESH_MODEL_OP_2(0x82, 0x96)
+#define BT_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x97)
+#define BT_MESH_MODEL_OP_LIGHT_LC_OM_STATUS BT_MESH_MODEL_OP_2(0x82, 0x98)
+#define BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET BT_MESH_MODEL_OP_2(0x82, 0x99)
+#define BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET BT_MESH_MODEL_OP_2(0x82, 0x9A)
+#define BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK BT_MESH_MODEL_OP_2(0x82, 0x9B)
+#define BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS BT_MESH_MODEL_OP_2(0x82, 0x9C)
+#define BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET BT_MESH_MODEL_OP_2(0x82, 0x9D)
+#define BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET BT_MESH_MODEL_OP_1(0x62)
+#define BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK BT_MESH_MODEL_OP_1(0x63)
+#define BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS BT_MESH_MODEL_OP_1(0x64)
+
+#endif /* __BT_MESH_MODEL_OP_H_ */
diff --git a/components/bt/ble_mesh/mesh_models/include/sensor_client.h b/components/bt/ble_mesh/mesh_models/include/sensor_client.h
new file mode 100644
index 0000000000..ebfd015c3a
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/include/sensor_client.h
@@ -0,0 +1,167 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Sensor Client Model APIs.
+ */
+
+#ifndef __BT_MESH_SENSOR_CLI_H
+#define __BT_MESH_SENSOR_CLI_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+#include "common.h"
+#include "bt_mesh_client_common.h"
+
+/* Sensor Client Model Context */
+extern const struct bt_mesh_model_op sensor_cli_op[];
+
+/** @def BT_MESH_MODEL_SENSOR_CLI
+ *
+ * Define a new sensor client model. Note that this API needs to
+ * be repeated for each element which the application wants to
+ * have a sensor client model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_sensor_cli.
+ *
+ * @return New sensor client model instance.
+ */
+#define BT_MESH_MODEL_SENSOR_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_SENSOR_CLI, \
+ sensor_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_sensor_client_t;
+typedef bt_mesh_internal_data_t sensor_internal_data_t;
+
+struct bt_mesh_sensor_descriptor_status {
+ struct net_buf_simple *descriptor; /* Sequence of 8-octet sensor descriptors (optional) */
+};
+
+struct bt_mesh_sensor_cadence_status {
+ u16_t property_id; /* Property for the sensor */
+ struct net_buf_simple *sensor_cadence_value; /* Value of sensor cadence state */
+};
+
+struct bt_mesh_sensor_settings_status {
+ u16_t sensor_property_id; /* Property ID identifying a sensor */
+ struct net_buf_simple *sensor_setting_property_ids; /* A sequence of N sensor setting property IDs (optional) */
+};
+
+struct bt_mesh_sensor_setting_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t sensor_property_id; /* Property ID identifying a sensor */
+ u16_t sensor_setting_property_id; /* Setting ID identifying a setting within a sensor */
+ u8_t sensor_setting_access; /* Read/Write access rights for the setting (optional) */
+ struct net_buf_simple *sensor_setting_raw; /* Raw value for the setting */
+};
+
+struct bt_mesh_sensor_status {
+ struct net_buf_simple *marshalled_sensor_data; /* Value of sensor data state (optional) */
+};
+
+struct bt_mesh_sensor_column_status {
+ u16_t property_id; /* Property identifying a sensor and the Y axis */
+ struct net_buf_simple *sensor_column_value; /* Left values of sensor column status */
+};
+
+struct bt_mesh_sensor_series_status {
+ u16_t property_id; /* Property identifying a sensor and the Y axis */
+ struct net_buf_simple *sensor_series_value; /* Left values of sensor series status */
+};
+
+struct bt_mesh_sensor_descriptor_get {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t property_id; /* Property ID for the sensor (optional) */
+};
+
+struct bt_mesh_sensor_cadence_get {
+ u16_t property_id; /* Property ID for the sensor */
+};
+
+struct bt_mesh_sensor_cadence_set {
+ u16_t property_id; /* Property ID for the sensor */
+ u8_t fast_cadence_period_divisor : 7, /* Divisor for the publish period */
+ status_trigger_type : 1; /* The unit and format of the Status Trigger Delta fields */
+ struct net_buf_simple *status_trigger_delta_down; /* Delta down value that triggers a status message */
+ struct net_buf_simple *status_trigger_delta_up; /* Delta up value that triggers a status message */
+ u8_t status_min_interval; /* Minimum interval between two consecutive Status messages */
+ struct net_buf_simple *fast_cadence_low; /* Low value for the fast cadence range */
+ struct net_buf_simple *fast_cadence_high; /* Fast value for the fast cadence range */
+};
+
+struct bt_mesh_sensor_settings_get {
+ u16_t sensor_property_id; /* Property ID for the sensor */
+};
+
+struct bt_mesh_sensor_setting_get {
+ u16_t sensor_property_id; /* Property ID identifying a sensor */
+ u16_t sensor_setting_property_id; /* Setting ID identifying a setting within a sensor */
+};
+
+struct bt_mesh_sensor_setting_set {
+ u16_t sensor_property_id; /* Property ID identifying a sensor */
+ u16_t sensor_setting_property_id; /* Setting ID identifying a setting within a sensor */
+ struct net_buf_simple *sensor_setting_raw; /* Raw value for the setting */
+};
+
+struct bt_mesh_sensor_get {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t property_id; /* Property ID for the sensor (optional) */
+};
+
+struct bt_mesh_sensor_column_get {
+ u16_t property_id; /* Property identifying a sensor */
+ struct net_buf_simple *raw_value_x; /* Raw value identifying a column */
+};
+
+struct bt_mesh_sensor_series_get {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t property_id; /* Property identifying a sensor */
+ struct net_buf_simple *raw_value_x1; /* Raw value identifying a starting column (optional) */
+ struct net_buf_simple *raw_value_x2; /* Raw value identifying a ending column (C.1) */
+};
+
+/**
+ * @brief This function is called to initialize sensor client model user_data.
+ *
+ * @param[in] model: Pointer to sensor client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_sensor_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to get sensor states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] get: Pointer of sensor get message value
+ * @param[out] status: Pointer of sensor status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_sensor_client_get_state(struct bt_mesh_common_param *common, void *get, void *status);
+
+/**
+ * @brief This function is called to set sensor states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] set: Pointer of sensor set message value
+ * @param[out] status: Pointer of sensor status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_sensor_client_set_state(struct bt_mesh_common_param *common, void *set, void *status);
+
+#endif /* __BT_MESH_SENSOR_CLI_H */
diff --git a/components/bt/ble_mesh/mesh_models/include/time_scene_client.h b/components/bt/ble_mesh/mesh_models/include/time_scene_client.h
new file mode 100644
index 0000000000..99ac6fa969
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/include/time_scene_client.h
@@ -0,0 +1,257 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+/** @file
+ * @brief Bluetooth Mesh Time and Scene Client Model APIs.
+ */
+
+#ifndef __BT_MESH_TIME_SCENE_CLI_H
+#define __BT_MESH_TIME_SCENE_CLI_H
+
+#include "mesh_access.h"
+#include "mesh_kernel.h"
+#include "common.h"
+#include "bt_mesh_client_common.h"
+
+/* Time scene client model common structure */
+typedef bt_mesh_client_common_t bt_mesh_time_scene_client_t;
+typedef bt_mesh_internal_data_t time_scene_internal_data_t;
+
+/* Time Client Model Context */
+extern const struct bt_mesh_model_op time_cli_op[];
+
+/** @def BT_MESH_MODEL_TIME_CLI
+ *
+ * Define a new time client model. Note that this API needs to
+ * be repeated for each element which the application wants to
+ * have a time model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_time_cli.
+ *
+ * @return New time client model instance.
+ */
+#define BT_MESH_MODEL_TIME_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_TIME_CLI, \
+ time_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_time_cli_t;
+
+struct bt_mesh_time_status {
+ u8_t tai_seconds[5]; /* The current TAI time in seconds */
+ u8_t sub_second; /* The sub-second time in units of 1/256 second */
+ u8_t uncertainty; /* The estimated uncertainty in 10-millisecond steps */
+ u16_t time_authority : 1; /* 0 = No Time Authority, 1 = Time Authority */
+ u16_t tai_utc_delta : 15; /* Current difference between TAI and UTC in seconds */
+ u8_t time_zone_offset; /* The local time zone offset in 15-minute increments */
+};
+
+struct bt_mesh_time_zone_status {
+ u8_t time_zone_offset_curr; /* Current local time zone offset */
+ u8_t time_zone_offset_new; /* Upcoming local time zone offset */
+ u8_t tai_zone_change[5]; /* TAI Seconds time of the upcoming Time Zone Offset change */
+};
+
+struct bt_mesh_tai_utc_delta_status {
+ u16_t tai_utc_delta_curr : 15; /* Current difference between TAI and UTC in seconds */
+ u16_t padding_1 : 1; /* Always 0b0. Other values are Prohibited. */
+ u16_t tai_utc_delta_new : 15; /* Upcoming difference between TAI and UTC in seconds */
+ u16_t padding_2 : 1; /* Always 0b0. Other values are Prohibited. */
+ u8_t tai_delta_change[5]; /* TAI Seconds time of the upcoming TAI-UTC Delta change */
+};
+
+struct bt_mesh_time_role_status {
+ u8_t time_role; /* The Time Role for the element */
+};
+
+struct bt_mesh_time_set {
+ u8_t tai_seconds[5]; /* The current TAI time in seconds */
+ u8_t sub_second; /* The sub-second time in units of 1/256 second */
+ u8_t uncertainty; /* The estimated uncertainty in 10-millisecond steps */
+ u16_t time_authority : 1; /* 0 = No Time Authority, 1 = Time Authority */
+ u16_t tai_utc_delta : 15; /* Current difference between TAI and UTC in seconds */
+ u8_t time_zone_offset; /* The local time zone offset in 15-minute increments */
+};
+
+struct bt_mesh_time_zone_set {
+ u8_t time_zone_offset_new; /* Upcoming local time zone offset */
+ u8_t tai_zone_change[5]; /* TAI Seconds time of the upcoming Time Zone Offset change */
+};
+
+struct bt_mesh_tai_utc_delta_set {
+ u16_t tai_utc_delta_new : 15; /* Upcoming difference between TAI and UTC in seconds */
+ u16_t padding : 1; /* Always 0b0. Other values are Prohibited. */
+ u8_t tai_delta_change[5]; /* TAI Seconds time of the upcoming TAI-UTC Delta change */
+};
+
+struct bt_mesh_time_role_set {
+ u8_t time_role; /* The Time Role for the element */
+};
+
+/* Scene Client Model Context */
+extern const struct bt_mesh_model_op scene_cli_op[];
+
+/** @def BT_MESH_MODEL_SCENE_CLI
+ *
+ * Define a new scene client model. Note that this API needs to
+ * be repeated for each element which the application wants to
+ * have a scene model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_scene_cli.
+ *
+ * @return New scene client model instance.
+ */
+#define BT_MESH_MODEL_SCENE_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_SCENE_CLI, \
+ scene_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_scene_cli_t;
+
+struct bt_mesh_scene_status {
+ bool op_en; /* Indicate whether optional parameters included */
+ u8_t status_code; /* Status code for the last operation */
+ u16_t current_scene; /* Scene Number of a current scene */
+ u16_t target_scene; /* Scene Number of a target scene (optional) */
+ u8_t remain_time; /* Time to complete state transition (C.1) */
+};
+
+struct bt_mesh_scene_register_status {
+ u8_t status_code; /* Status code for the previous operation */
+ u16_t current_scene; /* Scene Number of a current scene */
+ struct net_buf_simple *scenes; /* A list of scenes stored within an element */
+};
+
+struct bt_mesh_scene_store {
+ u16_t scene_number; /* The number of the scene to be stored */
+};
+
+struct bt_mesh_scene_recall {
+ bool op_en; /* Indicate whether optional parameters included */
+ u16_t scene_number; /* The number of the scene to be recalled */
+ u8_t tid; /* Transaction Identifier */
+ u8_t trans_time; /* Time to complete state transition (optional) */
+ u8_t delay; /* Indicate message execution delay (C.1) */
+};
+
+struct bt_mesh_scene_delete {
+ u16_t scene_number; /* The number of the scene to be deleted */
+};
+
+/* Scheduler Client Model Context */
+extern const struct bt_mesh_model_op scheduler_cli_op[];
+
+/** @def BT_MESH_MODEL_SCHEDULER_CLI
+ *
+ * Define a new scheduler client model. Note that this API needs to
+ * be repeated for each element which the application wants to
+ * have a scheduler model on.
+ * @param cli_pub Pointer to a unique struct bt_mesh_model_pub.
+ * @param cli_data Pointer to a unique struct bt_mesh_scheduler_cli.
+ *
+ * @return New scheduler client model instance.
+ */
+#define BT_MESH_MODEL_SCHEDULER_CLI(cli_pub, cli_data) \
+ BT_MESH_MODEL(BT_MESH_MODEL_ID_SCHEDULER_CLI, \
+ scheduler_cli_op, cli_pub, cli_data)
+
+typedef bt_mesh_client_common_t bt_mesh_scheduler_cli_t;
+
+struct bt_mesh_scheduler_status {
+ u16_t schedules; /* Bit field indicating defined Actions in the Schedule Register */
+};
+
+struct bt_mesh_scheduler_act_status {
+ u64_t index : 4; /* Enumerates (selects) a Schedule Register entry */
+ u64_t year : 7; /* Scheduled year for the action */
+ u64_t month : 12; /* Scheduled month for the action */
+ u64_t day : 5; /* Scheduled day of the month for the action */
+ u64_t hour : 5; /* Scheduled hour for the action */
+ u64_t minute : 6; /* Scheduled minute for the action */
+ u64_t second : 6; /* Scheduled second for the action */
+ u64_t day_of_week : 7; /* Schedule days of the week for the action */
+ u64_t action : 4; /* Action to be performed at the scheduled time */
+ u64_t trans_time : 8; /* Transition time for this action */
+ u16_t scene_number; /* Transition time for this action */
+};
+
+struct bt_mesh_scheduler_act_get {
+ u8_t index; /* Index of the Schedule Register entry to get */
+};
+
+struct bt_mesh_scheduler_act_set {
+ u64_t index : 4; /* Index of the Schedule Register entry to set */
+ u64_t year : 7; /* Scheduled year for the action */
+ u64_t month : 12; /* Scheduled month for the action */
+ u64_t day : 5; /* Scheduled day of the month for the action */
+ u64_t hour : 5; /* Scheduled hour for the action */
+ u64_t minute : 6; /* Scheduled minute for the action */
+ u64_t second : 6; /* Scheduled second for the action */
+ u64_t day_of_week : 7; /* Schedule days of the week for the action */
+ u64_t action : 4; /* Action to be performed at the scheduled time */
+ u64_t trans_time : 8; /* Transition time for this action */
+ u16_t scene_number; /* Transition time for this action */
+};
+
+/**
+ * @brief This function is called to initialize time client model user_data.
+ *
+ * @param[in] model: Pointer to time client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_time_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize scene client model user_data.
+ *
+ * @param[in] model: Pointer to scene client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_scene_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to initialize scheduler client model user_data.
+ *
+ * @param[in] model: Pointer to scheduler client model
+ * @param[in] primary: Whether belongs to primary element
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_scheduler_cli_init(struct bt_mesh_model *model, bool primary);
+
+/**
+ * @brief This function is called to get scene states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] get: Pointer of time scene get message value
+ * @param[out] status: Pointer of time scene status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_time_scene_client_get_state(struct bt_mesh_common_param *common, void *get, void *status);
+
+/**
+ * @brief This function is called to set scene states.
+ *
+ * @param[in] common: Message common information structure
+ * @param[in] set: Pointer of time scene set message value
+ * @param[out] status: Pointer of time scene status message value
+ *
+ * @return Zero-success, other-fail
+ */
+int bt_mesh_time_scene_client_set_state(struct bt_mesh_common_param *common, void *set, void *status);
+
+#endif /* __BT_MESH_TIME_SCENE_CLI_H */
diff --git a/components/bt/ble_mesh/mesh_models/light_client.c b/components/bt/ble_mesh/mesh_models/light_client.c
new file mode 100644
index 0000000000..48fe51a682
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/light_client.c
@@ -0,0 +1,1402 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+#include
+
+#include "osi/allocator.h"
+
+#include "mesh_types.h"
+#include "mesh_kernel.h"
+#include "mesh_trace.h"
+#include "mesh.h"
+
+#include "model_op.h"
+#include "common.h"
+#include "light_client.h"
+
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_light_client.h"
+
+/** The following are the macro definitions of lighting client
+ * model messages length, and a message is composed of three
+ * parts: Opcode + msg_value + MIC
+ */
+/* Light lightness client messages length */
+#define BT_MESH_LIGHT_LIGHTNESS_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_SET_MSG_LEN (2 + 5 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_LINEAR_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_LINEAR_SET_MSG_LEN (2 + 5 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_LAST_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_DEFAULT_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_DEFAULT_SET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_RANGE_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_LIGHTNESS_RANGE_SET_MSG_LEN (2 + 4 + 4)
+
+/* Light CTL client messages length */
+#define BT_MESH_LIGHT_CTL_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_CTL_SET_MSG_LEN (2 + 9 + 4)
+#define BT_MESH_LIGHT_CTL_TEMPERATURE_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_CTL_TEMPERATURE_SET_MSG_LEN (2 + 7 + 4)
+#define BT_MESH_LIGHT_CTL_TEMPERATURE_RANGE_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_CTL_TEMPERATURE_RANGE_SET_MSG_LEN (2 + 4 + 4)
+#define BT_MESH_LIGHT_CTL_DEFAULT_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_CTL_DEFAULT_SET_MSG_LEN (2 + 6 + 4)
+
+/* Light HSL client messages length */
+#define BT_MESH_LIGHT_HSL_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_HSL_SET_MSG_LEN (2 + 9 + 4)
+#define BT_MESH_LIGHT_HSL_TARGET_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_HSL_HUE_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_HSL_HUE_SET_MSG_LEN (2 + 5 + 4)
+#define BT_MESH_LIGHT_HSL_SATURATION_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_HSL_SATURATION_SET_MSG_LEN (2 + 5 + 4)
+#define BT_MESH_LIGHT_HSL_DEFAULT_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_HSL_DEFAULT_SET_MSG_LEN (2 + 6 + 4)
+#define BT_MESH_LIGHT_HSL_RANGE_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_LIGHT_HSL_RANGE_SET_MSG_LEN (2 + 8 + 4)
+
+/* Light xyL client messages length */
+#define BT_MESH_LIGHT_XYL_SET_MSG_LEN (2 + 9 + 4)
+#define BT_MESH_LIGHT_XYL_DEFAULT_SET_MSG_LEN (2 + 6 + 4)
+#define BT_MESH_LIGHT_XYL_RANGE_SET_MSG_LEN (2 + 8 + 4)
+
+/* Light LC client messages length */
+#define BT_MESH_LIGHT_LC_MODE_SET_MSG_LEN (2 + 1 + 4)
+#define BT_MESH_LIGHT_LC_OM_SET_MSG_LEN (2 + 1 + 4)
+#define BT_MESH_LIGHT_LC_LIGHT_ONOFF_SET_MSG_LEN (2 + 4 + 4)
+#define BT_MESH_LIGHT_LC_PROPERTY_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_LIGHT_LC_PROPERTY_SET_MSG_LEN /* variable */
+
+#define BT_MESH_LIGHT_GET_STATE_MSG_LEN (2 + 2 + 4)
+
+static const bt_mesh_client_op_pair_t light_op_pair[] = {
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET, BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_GET, BT_MESH_MODEL_OP_LIGHT_CTL_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_SET, BT_MESH_MODEL_OP_LIGHT_CTL_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET, BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET, BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET, BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET, BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET, BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET, BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_GET, BT_MESH_MODEL_OP_LIGHT_HSL_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_SET, BT_MESH_MODEL_OP_LIGHT_HSL_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET, BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_HUE_GET, BT_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET, BT_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET, BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET, BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET, BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET, BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET, BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET, BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_GET, BT_MESH_MODEL_OP_LIGHT_XYL_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_SET, BT_MESH_MODEL_OP_LIGHT_XYL_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET, BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET, BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET, BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET, BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET, BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_MODE_GET, BT_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET, BT_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_OM_GET, BT_MESH_MODEL_OP_LIGHT_LC_OM_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_OM_SET, BT_MESH_MODEL_OP_LIGHT_LC_OM_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET, BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET, BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET, BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS },
+ { BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET, BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS },
+};
+
+static void timeout_handler(struct k_work *work)
+{
+ bt_mesh_light_client_t *client = NULL;
+ light_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+
+ BT_WARN("Receive light status message timeout");
+
+ node = CONTAINER_OF(work, bt_mesh_client_node_t, timer.work);
+ if (!node || !node->ctx.model) {
+ BT_ERR("%s: node parameter is NULL", __func__);
+ return;
+ }
+
+ client = (bt_mesh_light_client_t *)node->ctx.model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (light_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: internal_data is NULL", __func__);
+ return;
+ }
+
+ bt_mesh_callback_light_status_to_btc(node->opcode, 0x03, node->ctx.model,
+ &node->ctx, NULL, 0);
+
+ bt_mesh_client_free_node(&internal->queue, node);
+
+ return;
+}
+
+static void light_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ bt_mesh_light_client_t *client = NULL;
+ light_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ u8_t *val = NULL;
+ u8_t evt = 0xFF;
+ u32_t rsp = 0;
+ size_t len = 0;
+
+ BT_DBG("%s: len %d, bytes %s", __func__, buf->len, bt_hex(buf->data, buf->len));
+
+ client = (bt_mesh_light_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (light_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: model internal_data is NULL", __func__);
+ return;
+ }
+
+ rsp = ctx->recv_op;
+
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS: {
+ struct bt_mesh_light_lightness_status *status = NULL;
+ if (buf->len != 2 && buf->len != 5) {
+ BT_ERR("Wrong Light lightness status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lightness_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_lightness = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_lightness = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lightness_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS: {
+ struct bt_mesh_light_lightness_linear_status *status = NULL;
+ if (buf->len != 2 && buf->len != 5) {
+ BT_ERR("Wrong Light lightness status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lightness_linear_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_lightness = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_lightness = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lightness_linear_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS: {
+ struct bt_mesh_light_lightness_last_status *status = NULL;
+ if (buf->len != 2) {
+ BT_ERR("Wrong Light last status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lightness_last_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->lightness = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lightness_last_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS: {
+ struct bt_mesh_light_lightness_default_status *status = NULL;
+ if (buf->len != 2) {
+ BT_ERR("Wrong Light default status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lightness_default_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->lightness = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lightness_default_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS: {
+ struct bt_mesh_light_lightness_range_status *status = NULL;
+ if (buf->len != 5) {
+ BT_ERR("Wrong Light last status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lightness_range_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->status_code = net_buf_simple_pull_u8(buf);
+ status->range_min = net_buf_simple_pull_le16(buf);
+ status->range_max = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lightness_range_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_STATUS: {
+ struct bt_mesh_light_ctl_status *status = NULL;
+ if (buf->len != 4 && buf->len != 9) {
+ BT_ERR("Wrong Light CTL status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_ctl_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_ctl_lightness = net_buf_simple_pull_le16(buf);
+ status->present_ctl_temperature = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_ctl_lightness = net_buf_simple_pull_le16(buf);
+ status->target_ctl_temperature = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_ctl_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS: {
+ struct bt_mesh_light_ctl_temperature_status *status = NULL;
+ if (buf->len != 4 && buf->len != 9) {
+ BT_ERR("Wrong Light CTL temperature status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_ctl_temperature_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_ctl_temperature = net_buf_simple_pull_le16(buf);
+ status->present_ctl_delta_uv = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_ctl_temperature = net_buf_simple_pull_le16(buf);
+ status->target_ctl_delta_uv = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_ctl_temperature_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS: {
+ struct bt_mesh_light_ctl_temperature_range_status *status = NULL;
+ if (buf->len != 5) {
+ BT_ERR("Wrong Light CTL temperature range status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_ctl_temperature_range_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->status_code = net_buf_simple_pull_u8(buf);
+ status->range_min = net_buf_simple_pull_le16(buf);
+ status->range_max = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_ctl_temperature_range_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS: {
+ struct bt_mesh_light_ctl_default_status *status = NULL;
+ if (buf->len != 6) {
+ BT_ERR("Wrong Light CTL default status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_ctl_default_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->lightness = net_buf_simple_pull_le16(buf);
+ status->temperature = net_buf_simple_pull_le16(buf);
+ status->delta_uv = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_ctl_default_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_STATUS: {
+ struct bt_mesh_light_hsl_status *status = NULL;
+ if (buf->len != 6 && buf->len != 7) {
+ BT_ERR("Wrong Light HSL status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_hsl_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->hsl_lightness = net_buf_simple_pull_le16(buf);
+ status->hsl_hue = net_buf_simple_pull_le16(buf);
+ status->hsl_saturation = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_hsl_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS: {
+ struct bt_mesh_light_hsl_target_status *status = NULL;
+ if (buf->len != 6 && buf->len != 7) {
+ BT_ERR("Wrong Light HSL target status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_hsl_target_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->hsl_lightness_target = net_buf_simple_pull_le16(buf);
+ status->hsl_hue_target = net_buf_simple_pull_le16(buf);
+ status->hsl_saturation_target = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_hsl_target_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS: {
+ struct bt_mesh_light_hsl_hue_status *status = NULL;
+ if (buf->len != 2 && buf->len != 5) {
+ BT_ERR("Wrong Light HSL Hue status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_hsl_hue_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_hue = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_hue = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_hsl_hue_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS: {
+ struct bt_mesh_light_hsl_saturation_status *status = NULL;
+ if (buf->len != 2 && buf->len != 5) {
+ BT_ERR("Wrong Light HSL saturation status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_hsl_saturation_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_saturation = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_saturation = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_hsl_saturation_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS: {
+ struct bt_mesh_light_hsl_default_status *status = NULL;
+ if (buf->len != 6) {
+ BT_ERR("Wrong Light HSL default status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_hsl_default_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->lightness = net_buf_simple_pull_le16(buf);
+ status->hue = net_buf_simple_pull_le16(buf);
+ status->saturation = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_hsl_default_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS: {
+ struct bt_mesh_light_hsl_range_status *status = NULL;
+ if (buf->len != 9) {
+ BT_ERR("Wrong Light HSL range status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_hsl_range_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->status_code = net_buf_simple_pull_u8(buf);
+ status->hue_range_min = net_buf_simple_pull_le16(buf);
+ status->hue_range_max = net_buf_simple_pull_le16(buf);
+ status->saturation_range_min = net_buf_simple_pull_le16(buf);
+ status->saturation_range_max = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_hsl_range_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_STATUS: {
+ struct bt_mesh_light_xyl_status *status = NULL;
+ if (buf->len != 6 && buf->len != 7) {
+ BT_ERR("Wrong Light xyL status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_xyl_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->xyl_lightness = net_buf_simple_pull_le16(buf);
+ status->xyl_x = net_buf_simple_pull_le16(buf);
+ status->xyl_y = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_xyl_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS: {
+ struct bt_mesh_light_xyl_target_status *status = NULL;
+ if (buf->len != 6 && buf->len != 7) {
+ BT_ERR("Wrong Light xyL Target status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_xyl_target_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->target_xyl_lightness = net_buf_simple_pull_le16(buf);
+ status->target_xyl_x = net_buf_simple_pull_le16(buf);
+ status->target_xyl_y = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_xyl_target_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS: {
+ struct bt_mesh_light_xyl_default_status *status = NULL;
+ if (buf->len != 6) {
+ BT_ERR("Wrong Light xyL Default status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_xyl_default_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->lightness = net_buf_simple_pull_le16(buf);
+ status->xyl_x = net_buf_simple_pull_le16(buf);
+ status->xyl_y = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_xyl_default_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS: {
+ struct bt_mesh_light_xyl_range_status *status = NULL;
+ if (buf->len != 9) {
+ BT_ERR("Wrong Light xyL Range status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_xyl_range_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->status_code = net_buf_simple_pull_u8(buf);
+ status->xyl_x_range_min = net_buf_simple_pull_le16(buf);
+ status->xyl_x_range_max = net_buf_simple_pull_le16(buf);
+ status->xyl_y_range_min = net_buf_simple_pull_le16(buf);
+ status->xyl_y_range_max = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_xyl_range_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS: {
+ struct bt_mesh_light_lc_mode_status *status = NULL;
+ if (buf->len != 1) {
+ BT_ERR("Wrong Light LC Mode status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lc_mode_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->mode = net_buf_simple_pull_u8(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lc_mode_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_STATUS: {
+ struct bt_mesh_light_lc_om_status *status = NULL;
+ if (buf->len != 1) {
+ BT_ERR("Wrong Light LC OM status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lc_om_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->mode = net_buf_simple_pull_u8(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lc_om_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS: {
+ struct bt_mesh_light_lc_light_onoff_status *status = NULL;
+ if (buf->len != 1 && buf->len != 3) {
+ BT_ERR("Wrong Light LC Light OnOff status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_light_lc_light_onoff_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->present_light_onoff = net_buf_simple_pull_u8(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_light_onoff = net_buf_simple_pull_u8(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lc_light_onoff_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS: {
+ struct bt_mesh_light_lc_property_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_light_lc_property_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->light_lc_property_id = net_buf_simple_pull_le16(buf);
+ status->light_lc_property_value = bt_mesh_alloc_buf(buf->len);
+ if (!status->light_lc_property_value) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->light_lc_property_value, 0);
+ net_buf_simple_add_mem(status->light_lc_property_value, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_light_lc_property_status);
+ break;
+ }
+ default:
+ BT_ERR("Not a lighting status message opcode");
+ return;
+ }
+
+ buf->data = val;
+ buf->len = len;
+ node = bt_mesh_is_model_message_publish(model, ctx, buf, true);
+ if (!node) {
+ BT_DBG("Unexpected light status message 0x%x", rsp);
+ } else {
+ switch (node->opcode) {
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET:
+ evt = 0x00;
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_SET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
+ evt = 0x01;
+ break;
+ default:
+ break;
+ }
+
+ bt_mesh_callback_light_status_to_btc(node->opcode, evt, model, ctx, val, len);
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&internal->queue, node);
+ }
+
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS: {
+ struct bt_mesh_light_lc_property_status *status;
+ status = (struct bt_mesh_light_lc_property_status *)val;
+ bt_mesh_free_buf(status->light_lc_property_value);
+ break;
+ }
+ default:
+ break;
+ }
+
+ osi_free(val);
+
+ return;
+}
+
+const struct bt_mesh_model_op light_lightness_cli_op[] = {
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_STATUS, 2, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_STATUS, 2, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_STATUS, 2, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_STATUS, 2, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_STATUS, 5, light_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op light_ctl_cli_op[] = {
+ { BT_MESH_MODEL_OP_LIGHT_CTL_STATUS, 4, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_STATUS, 4, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_STATUS, 5, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_STATUS, 6, light_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op light_hsl_cli_op[] = {
+ { BT_MESH_MODEL_OP_LIGHT_HSL_STATUS, 6, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_STATUS, 6, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_HUE_STATUS, 2, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_STATUS, 2, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_STATUS, 6, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_STATUS, 9, light_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op light_xyl_cli_op[] = {
+ { BT_MESH_MODEL_OP_LIGHT_XYL_STATUS, 6, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_STATUS, 6, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_STATUS, 6, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_STATUS, 9, light_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op light_lc_cli_op[] = {
+ { BT_MESH_MODEL_OP_LIGHT_LC_MODE_STATUS, 1, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_LC_OM_STATUS, 1, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_STATUS, 1, light_status },
+ { BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_STATUS, 2, light_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+static int light_get_state(struct bt_mesh_common_param *common, void *value)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(BT_MESH_LIGHT_GET_STATE_MSG_LEN);
+ int err;
+
+ bt_mesh_model_msg_init(msg, common->opcode);
+
+ if (value) {
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET: {
+ struct bt_mesh_light_lc_property_get *get;
+ get = (struct bt_mesh_light_lc_property_get *)value;
+ net_buf_simple_add_le16(msg, get->light_lc_property_id);
+ break;
+ }
+ default:
+ BT_DBG("This lighting message should be sent with NULL get pointer");
+ break;
+ }
+ }
+
+ err = bt_mesh_client_send_msg(common->model, common->opcode, &common->ctx, msg,
+ timeout_handler, common->msg_timeout, true,
+ common->cb, common->cb_data);
+ if (err) {
+ BT_ERR("Light get message send failed (err %d)", err);
+ }
+
+ return err;
+}
+
+static int light_set_state(struct bt_mesh_common_param *common,
+ void *value, u16_t value_len, bool need_ack)
+{
+ struct net_buf_simple *msg = NULL;
+ int err;
+
+ msg = bt_mesh_alloc_buf(value_len);
+ if (!msg) {
+ BT_ERR("Light set allocate memory fail");
+ return -ENOMEM;
+ }
+
+ bt_mesh_model_msg_init(msg, common->opcode);
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK: {
+ struct bt_mesh_light_lightness_set *set;
+ set = (struct bt_mesh_light_lightness_set *)value;
+ net_buf_simple_add_le16(msg, set->lightness);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK: {
+ struct bt_mesh_light_lightness_linear_set *set;
+ set = (struct bt_mesh_light_lightness_linear_set *)value;
+ net_buf_simple_add_le16(msg, set->lightness);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK: {
+ struct bt_mesh_light_lightness_default_set *set;
+ set = (struct bt_mesh_light_lightness_default_set *)value;
+ net_buf_simple_add_le16(msg, set->lightness);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK: {
+ struct bt_mesh_light_lightness_range_set *set;
+ set = (struct bt_mesh_light_lightness_range_set *)value;
+ net_buf_simple_add_le16(msg, set->range_min);
+ net_buf_simple_add_le16(msg, set->range_max);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK: {
+ struct bt_mesh_light_ctl_set *set;
+ set = (struct bt_mesh_light_ctl_set *)value;
+ net_buf_simple_add_le16(msg, set->ctl_lightness);
+ net_buf_simple_add_le16(msg, set->ctl_temperature);
+ net_buf_simple_add_le16(msg, set->ctl_delta_uv);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK: {
+ struct bt_mesh_light_ctl_temperature_set *set;
+ set = (struct bt_mesh_light_ctl_temperature_set *)value;
+ net_buf_simple_add_le16(msg, set->ctl_temperature);
+ net_buf_simple_add_le16(msg, set->ctl_delta_uv);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK: {
+ struct bt_mesh_light_ctl_temperature_range_set *set;
+ set = (struct bt_mesh_light_ctl_temperature_range_set *)value;
+ net_buf_simple_add_le16(msg, set->range_min);
+ net_buf_simple_add_le16(msg, set->range_max);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK: {
+ struct bt_mesh_light_ctl_default_set *set;
+ set = (struct bt_mesh_light_ctl_default_set *)value;
+ net_buf_simple_add_le16(msg, set->lightness);
+ net_buf_simple_add_le16(msg, set->temperature);
+ net_buf_simple_add_le16(msg, set->delta_uv);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK: {
+ struct bt_mesh_light_hsl_set *set;
+ set = (struct bt_mesh_light_hsl_set *)value;
+ net_buf_simple_add_le16(msg, set->hsl_lightness);
+ net_buf_simple_add_le16(msg, set->hsl_hue);
+ net_buf_simple_add_le16(msg, set->hsl_saturation);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK: {
+ struct bt_mesh_light_hsl_hue_set *set;
+ set = (struct bt_mesh_light_hsl_hue_set *)value;
+ net_buf_simple_add_le16(msg, set->hue);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK: {
+ struct bt_mesh_light_hsl_saturation_set *set;
+ set = (struct bt_mesh_light_hsl_saturation_set *)value;
+ net_buf_simple_add_le16(msg, set->saturation);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK: {
+ struct bt_mesh_light_hsl_default_set *set;
+ set = (struct bt_mesh_light_hsl_default_set *)value;
+ net_buf_simple_add_le16(msg, set->lightness);
+ net_buf_simple_add_le16(msg, set->hue);
+ net_buf_simple_add_le16(msg, set->saturation);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK: {
+ struct bt_mesh_light_hsl_range_set *set;
+ set = (struct bt_mesh_light_hsl_range_set *)value;
+ net_buf_simple_add_le16(msg, set->hue_range_min);
+ net_buf_simple_add_le16(msg, set->hue_range_max);
+ net_buf_simple_add_le16(msg, set->saturation_range_min);
+ net_buf_simple_add_le16(msg, set->saturation_range_max);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_SET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK: {
+ struct bt_mesh_light_xyl_set *set;
+ set = (struct bt_mesh_light_xyl_set *)value;
+ net_buf_simple_add_le16(msg, set->xyl_lightness);
+ net_buf_simple_add_le16(msg, set->xyl_x);
+ net_buf_simple_add_le16(msg, set->xyl_y);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK: {
+ struct bt_mesh_light_xyl_default_set *set;
+ set = (struct bt_mesh_light_xyl_default_set *)value;
+ net_buf_simple_add_le16(msg, set->lightness);
+ net_buf_simple_add_le16(msg, set->xyl_x);
+ net_buf_simple_add_le16(msg, set->xyl_y);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK: {
+ struct bt_mesh_light_xyl_range_set *set;
+ set = (struct bt_mesh_light_xyl_range_set *)value;
+ net_buf_simple_add_le16(msg, set->xyl_x_range_min);
+ net_buf_simple_add_le16(msg, set->xyl_x_range_max);
+ net_buf_simple_add_le16(msg, set->xyl_y_range_min);
+ net_buf_simple_add_le16(msg, set->xyl_y_range_max);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK: {
+ struct bt_mesh_light_lc_mode_set *set;
+ set = (struct bt_mesh_light_lc_mode_set *)value;
+ net_buf_simple_add_u8(msg, set->mode);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK: {
+ struct bt_mesh_light_lc_om_set *set;
+ set = (struct bt_mesh_light_lc_om_set *)value;
+ net_buf_simple_add_u8(msg, set->mode);
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK: {
+ struct bt_mesh_light_lc_light_onoff_set *set;
+ set = (struct bt_mesh_light_lc_light_onoff_set *)value;
+ net_buf_simple_add_u8(msg, set->light_onoff);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK: {
+ struct bt_mesh_light_lc_property_set *set;
+ set = (struct bt_mesh_light_lc_property_set *)value;
+ net_buf_simple_add_le16(msg, set->light_lc_property_id);
+ net_buf_simple_add_mem(msg, set->light_lc_property_value->data, set->light_lc_property_value->len);
+ break;
+ }
+ default:
+ BT_ERR("Not a lighting client model set message opcode");
+ err = -EINVAL;
+ goto end;
+ }
+
+ err = bt_mesh_client_send_msg(common->model, common->opcode, &common->ctx, msg,
+ timeout_handler, common->msg_timeout, need_ack,
+ common->cb, common->cb_data);
+ if (err) {
+ BT_ERR("Light set message send failed (err %d)", err);
+ }
+
+end:
+ bt_mesh_free_buf(msg);
+
+ return err;
+}
+
+int bt_mesh_light_client_get_state(struct bt_mesh_common_param *common, void *get, void *status)
+{
+ bt_mesh_light_client_t *client = NULL;
+
+ if (!common || !common->model) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_light_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LAST_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_TARGET_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_TARGET_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_GET:
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_GET:
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_GET:
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_GET:
+ if (!get) {
+ BT_ERR("Lighting lc_property_get is NULL");
+ return -EINVAL;
+ }
+ break;
+ default:
+ BT_ERR("Not a lighting get message opcode");
+ return -EINVAL;
+ }
+
+ return light_get_state(common, get);
+}
+
+int bt_mesh_light_client_set_state(struct bt_mesh_common_param *common, void *set, void *status)
+{
+ bt_mesh_light_client_t *client = NULL;
+ u16_t length = 0;
+ bool need_ack = false;
+
+ if (!common || !common->model || !set) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_light_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_SET_UNACK: {
+ struct bt_mesh_light_lightness_set *value;
+ value = (struct bt_mesh_light_lightness_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light lightness set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_LIGHTNESS_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_LINEAR_SET_UNACK: {
+ struct bt_mesh_light_lightness_linear_set *value;
+ value = (struct bt_mesh_light_lightness_linear_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light lightness linear set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_LIGHTNESS_LINEAR_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_DEFAULT_SET_UNACK:
+ length = BT_MESH_LIGHT_LIGHTNESS_DEFAULT_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LIGHTNESS_RANGE_SET_UNACK: {
+ struct bt_mesh_light_lightness_range_set *value;
+ value = (struct bt_mesh_light_lightness_range_set *)set;
+ if (value->range_min > value->range_max) {
+ BT_ERR("Light lightness range set range min is greater than range max");
+ return -EINVAL;
+ }
+ length = BT_MESH_LIGHT_LIGHTNESS_RANGE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_CTL_SET_UNACK: {
+ struct bt_mesh_light_ctl_set *value;
+ value = (struct bt_mesh_light_ctl_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light ctl set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_CTL_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_SET_UNACK: {
+ struct bt_mesh_light_ctl_temperature_set *value;
+ value = (struct bt_mesh_light_ctl_temperature_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light ctl temperature set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_CTL_TEMPERATURE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_CTL_TEMPERATURE_RANGE_SET_UNACK: {
+ struct bt_mesh_light_ctl_temperature_range_set *value;
+ value = (struct bt_mesh_light_ctl_temperature_range_set *)set;
+ if (value->range_min > value->range_max) {
+ BT_ERR("Light ctl temperature range set range min is greater than range max");
+ return -EINVAL;
+ }
+ length = BT_MESH_LIGHT_CTL_TEMPERATURE_RANGE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_CTL_DEFAULT_SET_UNACK:
+ length = BT_MESH_LIGHT_CTL_DEFAULT_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SET_UNACK: {
+ struct bt_mesh_light_hsl_set *value;
+ value = (struct bt_mesh_light_hsl_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light hsl set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_HSL_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_HSL_HUE_SET_UNACK: {
+ struct bt_mesh_light_hsl_hue_set *value;
+ value = (struct bt_mesh_light_hsl_hue_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light hsl hue set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_HSL_HUE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_HSL_SATURATION_SET_UNACK: {
+ struct bt_mesh_light_hsl_saturation_set *value;
+ value = (struct bt_mesh_light_hsl_saturation_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light hsl saturation set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_HSL_SATURATION_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_HSL_DEFAULT_SET_UNACK:
+ length = BT_MESH_LIGHT_HSL_DEFAULT_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_HSL_RANGE_SET_UNACK: {
+ struct bt_mesh_light_hsl_range_set *value;
+ value = (struct bt_mesh_light_hsl_range_set *)set;
+ if (value->hue_range_min > value->hue_range_max ||
+ value->saturation_range_min > value->saturation_range_max) {
+ BT_ERR("Light hsl range set range min is greater than range max");
+ return -EINVAL;
+ }
+ length = BT_MESH_LIGHT_HSL_RANGE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_XYL_SET_UNACK: {
+ struct bt_mesh_light_xyl_set *value;
+ value = (struct bt_mesh_light_xyl_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light xyl set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_XYL_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_XYL_DEFAULT_SET_UNACK:
+ length = BT_MESH_LIGHT_XYL_DEFAULT_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_XYL_RANGE_SET_UNACK: {
+ struct bt_mesh_light_xyl_range_set *value;
+ value = (struct bt_mesh_light_xyl_range_set *)set;
+ if (value->xyl_x_range_min > value->xyl_x_range_max ||
+ value->xyl_y_range_min > value->xyl_y_range_max) {
+ BT_ERR("Light xyl range set range min is greater than range max");
+ return -EINVAL;
+ }
+ length = BT_MESH_LIGHT_XYL_RANGE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LC_MODE_SET_UNACK:
+ length = BT_MESH_LIGHT_LC_MODE_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LC_OM_SET_UNACK:
+ length = BT_MESH_LIGHT_LC_OM_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LC_LIGHT_ONOFF_SET_UNACK: {
+ struct bt_mesh_light_lc_light_onoff_set *value;
+ value = (struct bt_mesh_light_lc_light_onoff_set *)set;
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Light lc light onoff set transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_LIGHT_LC_LIGHT_ONOFF_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_LIGHT_LC_PROPERTY_SET_UNACK: {
+ struct bt_mesh_light_lc_property_set *value;
+ value = (struct bt_mesh_light_lc_property_set *)set;
+ if (!value->light_lc_property_value) {
+ BT_ERR("Lighting light_lc_property_value is NULL");
+ return -EINVAL;
+ }
+ length = (1 + 2 + value->light_lc_property_value->len + 4);
+ break;
+ }
+ default:
+ BT_ERR("Not a lighting set message opcode");
+ return -EINVAL;
+ }
+
+ return light_set_state(common, set, length, need_ack);
+}
+
+static int light_client_init(struct bt_mesh_model *model, bool primary)
+{
+ bt_mesh_light_client_t *client = NULL;
+ light_internal_data_t *internal = NULL;
+
+ BT_DBG("primary %u", primary);
+
+ if (!model) {
+ BT_ERR("Light client model is NULL");
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_light_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("Light client model user_data is NULL");
+ return -EINVAL;
+ }
+
+ /* TODO: call osi_free() when deinit function is invoked*/
+ internal = osi_calloc(sizeof(light_internal_data_t));
+ if (!internal) {
+ BT_ERR("Allocate memory for generic onoff internal data fail");
+ return -ENOMEM;
+ }
+
+ sys_slist_init(&internal->queue);
+
+ client->model = model;
+ client->op_pair_size = ARRAY_SIZE(light_op_pair);
+ client->op_pair = light_op_pair;
+ client->internal_data = internal;
+
+ return 0;
+}
+
+int bt_mesh_light_lightness_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return light_client_init(model, primary);
+}
+
+int bt_mesh_light_ctl_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return light_client_init(model, primary);
+}
+
+int bt_mesh_light_hsl_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return light_client_init(model, primary);
+}
+
+int bt_mesh_light_xyl_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return light_client_init(model, primary);
+}
+
+int bt_mesh_light_lc_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return light_client_init(model, primary);
+}
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_models/sensor_client.c b/components/bt/ble_mesh/mesh_models/sensor_client.c
new file mode 100644
index 0000000000..4bd7a25ae4
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/sensor_client.c
@@ -0,0 +1,624 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+#include
+
+#include "osi/allocator.h"
+
+#include "mesh_types.h"
+#include "mesh_kernel.h"
+#include "mesh_trace.h"
+#include "mesh.h"
+
+#include "model_op.h"
+#include "common.h"
+#include "sensor_client.h"
+
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_sensor_client.h"
+
+/** The following are the macro definitions of sensor client
+ * model messages length, and a message is composed of three
+ * parts: Opcode + msg_value + MIC
+ */
+/* Sensor client messages length */
+#define BT_MESH_SENSOR_DESCRIPTOR_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_SENSOR_CADENCE_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_SENSOR_CADENCE_SET_MSG_LEN /* variable */
+#define BT_MESH_SENSOR_SETTINGS_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_SENSOR_SETTING_GET_MSG_LEN (2 + 4 + 4)
+#define BT_MESH_SENSOR_SETTING_SET_MSG_LEN /* variable */
+#define BT_MESH_SENSOR_GET_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_SENSOR_COLUMN_GET_MSG_LEN /* variable */
+#define BT_MESH_SENSOR_SERIES_GET_MSG_LEN /* variable */
+
+static const bt_mesh_client_op_pair_t sensor_op_pair[] = {
+ { BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET, BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_CADENCE_GET, BT_MESH_MODEL_OP_SENSOR_CADENCE_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_CADENCE_SET, BT_MESH_MODEL_OP_SENSOR_CADENCE_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_SETTINGS_GET, BT_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_SETTING_GET, BT_MESH_MODEL_OP_SENSOR_SETTING_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_SETTING_SET, BT_MESH_MODEL_OP_SENSOR_SETTING_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_GET, BT_MESH_MODEL_OP_SENSOR_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_COLUMN_GET, BT_MESH_MODEL_OP_SENSOR_COLUMN_STATUS },
+ { BT_MESH_MODEL_OP_SENSOR_SERIES_GET, BT_MESH_MODEL_OP_SENSOR_SERIES_STATUS },
+};
+
+static void timeout_handler(struct k_work *work)
+{
+ bt_mesh_sensor_client_t *client = NULL;
+ sensor_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+
+ BT_WARN("Receive sensor status message timeout");
+
+ node = CONTAINER_OF(work, bt_mesh_client_node_t, timer.work);
+ if (!node || !node->ctx.model) {
+ BT_ERR("%s: node parameter is NULL", __func__);
+ return;
+ }
+
+ client = (bt_mesh_sensor_client_t *)node->ctx.model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (sensor_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: internal_data is NULL", __func__);
+ return;
+ }
+
+ bt_mesh_callback_sensor_status_to_btc(node->opcode, 0x03, node->ctx.model,
+ &node->ctx, NULL, 0);
+
+ bt_mesh_client_free_node(&internal->queue, node);
+
+ return;
+}
+
+static void sensor_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ bt_mesh_sensor_client_t *client = NULL;
+ sensor_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ u8_t *val = NULL;
+ u8_t evt = 0xFF;
+ u32_t rsp = 0;
+ size_t len = 0;
+
+ BT_DBG("%s: len %d, bytes %s", __func__, buf->len, bt_hex(buf->data, buf->len));
+
+ client = (bt_mesh_sensor_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (sensor_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: model internal_data is NULL", __func__);
+ return;
+ }
+
+ rsp = ctx->recv_op;
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS: {
+ struct bt_mesh_sensor_descriptor_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_sensor_descriptor_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->descriptor = bt_mesh_alloc_buf(buf->len);
+ if (!status->descriptor) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->descriptor, 0);
+ net_buf_simple_add_mem(status->descriptor, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_sensor_descriptor_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_STATUS: {
+ struct bt_mesh_sensor_cadence_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_sensor_cadence_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->property_id = net_buf_simple_pull_le16(buf);
+ status->sensor_cadence_value = bt_mesh_alloc_buf(buf->len);
+ if (!status->sensor_cadence_value) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->sensor_cadence_value, 0);
+ net_buf_simple_add_mem(status->sensor_cadence_value, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_sensor_cadence_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS: {
+ struct bt_mesh_sensor_settings_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_sensor_settings_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->sensor_property_id = net_buf_simple_pull_le16(buf);
+ status->sensor_setting_property_ids = bt_mesh_alloc_buf(buf->len);
+ if (!status->sensor_setting_property_ids) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->sensor_setting_property_ids, 0);
+ net_buf_simple_add_mem(status->sensor_setting_property_ids, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_sensor_settings_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_STATUS: {
+ struct bt_mesh_sensor_setting_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_sensor_setting_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->sensor_property_id = net_buf_simple_pull_le16(buf);
+ status->sensor_setting_property_id = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->sensor_setting_access = net_buf_simple_pull_u8(buf);
+ status->sensor_setting_raw = bt_mesh_alloc_buf(buf->len);
+ if (!status->sensor_setting_raw) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->sensor_setting_raw, 0);
+ net_buf_simple_add_mem(status->sensor_setting_raw, buf->data, buf->len);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_sensor_setting_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_STATUS: {
+ struct bt_mesh_sensor_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_sensor_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->marshalled_sensor_data = bt_mesh_alloc_buf(buf->len);
+ if (!status->marshalled_sensor_data) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->marshalled_sensor_data, 0);
+ net_buf_simple_add_mem(status->marshalled_sensor_data, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_sensor_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_COLUMN_STATUS: {
+ struct bt_mesh_sensor_column_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_sensor_column_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->property_id = net_buf_simple_pull_le16(buf);
+ status->sensor_column_value = bt_mesh_alloc_buf(buf->len);
+ if (!status->sensor_column_value) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->sensor_column_value, 0);
+ net_buf_simple_add_mem(status->sensor_column_value, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_sensor_column_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SERIES_STATUS: {
+ struct bt_mesh_sensor_series_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_sensor_series_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->property_id = net_buf_simple_pull_le16(buf);
+ status->sensor_series_value = bt_mesh_alloc_buf(buf->len);
+ if (!status->sensor_series_value) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->sensor_series_value, 0);
+ net_buf_simple_add_mem(status->sensor_series_value, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_sensor_series_status);
+ break;
+ }
+ default:
+ BT_ERR("Not a sensor status message opcode");
+ return;
+ }
+
+ buf->data = val;
+ buf->len = len;
+ node = bt_mesh_is_model_message_publish(model, ctx, buf, true);
+ if (!node) {
+ BT_DBG("Unexpected sensor status message 0x%x", rsp);
+ } else {
+ switch (node->opcode) {
+ case BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET:
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_GET:
+ case BT_MESH_MODEL_OP_SENSOR_SETTINGS_GET:
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_GET:
+ case BT_MESH_MODEL_OP_SENSOR_GET:
+ case BT_MESH_MODEL_OP_SENSOR_COLUMN_GET:
+ case BT_MESH_MODEL_OP_SENSOR_SERIES_GET:
+ evt = 0x00;
+ break;
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_SET:
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_SET:
+ evt = 0x01;
+ break;
+ default:
+ break;
+ }
+
+ bt_mesh_callback_sensor_status_to_btc(node->opcode, evt, model, ctx, val, len);
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&internal->queue, node);
+ }
+
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS: {
+ struct bt_mesh_sensor_descriptor_status *status;
+ status = (struct bt_mesh_sensor_descriptor_status *)val;
+ bt_mesh_free_buf(status->descriptor);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_STATUS: {
+ struct bt_mesh_sensor_cadence_status *status;
+ status = (struct bt_mesh_sensor_cadence_status *)val;
+ bt_mesh_free_buf(status->sensor_cadence_value);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS: {
+ struct bt_mesh_sensor_settings_status *status;
+ status = (struct bt_mesh_sensor_settings_status *)val;
+ bt_mesh_free_buf(status->sensor_setting_property_ids);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_STATUS: {
+ struct bt_mesh_sensor_setting_status *status;
+ status = (struct bt_mesh_sensor_setting_status *)val;
+ bt_mesh_free_buf(status->sensor_setting_raw);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_STATUS: {
+ struct bt_mesh_sensor_status *status;
+ status = (struct bt_mesh_sensor_status *)val;
+ bt_mesh_free_buf(status->marshalled_sensor_data);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_COLUMN_STATUS: {
+ struct bt_mesh_sensor_column_status *status;
+ status = (struct bt_mesh_sensor_column_status *)val;
+ bt_mesh_free_buf(status->sensor_column_value);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SERIES_STATUS: {
+ struct bt_mesh_sensor_series_status *status;
+ status = (struct bt_mesh_sensor_series_status *)val;
+ bt_mesh_free_buf(status->sensor_series_value);
+ break;
+ }
+ default:
+ break;
+ }
+
+ osi_free(val);
+
+ return;
+}
+
+const struct bt_mesh_model_op sensor_cli_op[] = {
+ { BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_STATUS, 0, sensor_status },
+ { BT_MESH_MODEL_OP_SENSOR_CADENCE_STATUS, 2, sensor_status },
+ { BT_MESH_MODEL_OP_SENSOR_SETTINGS_STATUS, 2, sensor_status },
+ { BT_MESH_MODEL_OP_SENSOR_SETTING_STATUS, 4, sensor_status },
+ { BT_MESH_MODEL_OP_SENSOR_STATUS, 0, sensor_status },
+ { BT_MESH_MODEL_OP_SENSOR_COLUMN_STATUS, 2, sensor_status },
+ { BT_MESH_MODEL_OP_SENSOR_SERIES_STATUS, 2, sensor_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+static int sensor_act_state(struct bt_mesh_common_param *common,
+ void *value, u16_t value_len, bool need_ack)
+{
+ struct net_buf_simple *msg = NULL;
+ int err;
+
+ msg = bt_mesh_alloc_buf(value_len);
+ if (!msg) {
+ BT_ERR("Sensor allocate memory fail");
+ return -ENOMEM;
+ }
+
+ bt_mesh_model_msg_init(msg, common->opcode);
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET: {
+ struct bt_mesh_sensor_descriptor_get *act;
+ act = (struct bt_mesh_sensor_descriptor_get *)value;
+ if (act->op_en) {
+ net_buf_simple_add_le16(msg, act->property_id);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_GET: {
+ struct bt_mesh_sensor_cadence_get *act;
+ act = (struct bt_mesh_sensor_cadence_get *)value;
+ net_buf_simple_add_le16(msg, act->property_id);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_SET:
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_SET_UNACK: {
+ struct bt_mesh_sensor_cadence_set *act;
+ act = (struct bt_mesh_sensor_cadence_set *)value;
+ net_buf_simple_add_le16(msg, act->property_id);
+ net_buf_simple_add_u8(msg, act->status_trigger_type << 7 | act->fast_cadence_period_divisor);
+ net_buf_simple_add_mem(msg, act->status_trigger_delta_down->data, act->status_trigger_delta_down->len);
+ net_buf_simple_add_mem(msg, act->status_trigger_delta_up->data, act->status_trigger_delta_up->len);
+ net_buf_simple_add_u8(msg, act->status_min_interval);
+ net_buf_simple_add_mem(msg, act->fast_cadence_low->data, act->fast_cadence_low->len);
+ net_buf_simple_add_mem(msg, act->fast_cadence_high->data, act->fast_cadence_high->len);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTINGS_GET: {
+ struct bt_mesh_sensor_settings_get *act;
+ act = (struct bt_mesh_sensor_settings_get *)value;
+ net_buf_simple_add_le16(msg, act->sensor_property_id);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_GET: {
+ struct bt_mesh_sensor_setting_get *act;
+ act = (struct bt_mesh_sensor_setting_get *)value;
+ net_buf_simple_add_le16(msg, act->sensor_property_id);
+ net_buf_simple_add_le16(msg, act->sensor_setting_property_id);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_SET:
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_SET_UNACK: {
+ struct bt_mesh_sensor_setting_set *act;
+ act = (struct bt_mesh_sensor_setting_set *)value;
+ net_buf_simple_add_le16(msg, act->sensor_property_id);
+ net_buf_simple_add_le16(msg, act->sensor_setting_property_id);
+ net_buf_simple_add_mem(msg, act->sensor_setting_raw->data, act->sensor_setting_raw->len);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_GET: {
+ struct bt_mesh_sensor_get *act;
+ act = (struct bt_mesh_sensor_get *)value;
+ if (act->op_en) {
+ net_buf_simple_add_le16(msg, act->property_id);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_COLUMN_GET: {
+ struct bt_mesh_sensor_column_get *act;
+ act = (struct bt_mesh_sensor_column_get *)value;
+ net_buf_simple_add_le16(msg, act->property_id);
+ net_buf_simple_add_mem(msg, act->raw_value_x->data, act->raw_value_x->len);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SERIES_GET: {
+ struct bt_mesh_sensor_series_get *act;
+ act = (struct bt_mesh_sensor_series_get *)value;
+ net_buf_simple_add_le16(msg, act->property_id);
+ if (act->op_en) {
+ net_buf_simple_add_mem(msg, act->raw_value_x1->data, act->raw_value_x1->len);
+ net_buf_simple_add_mem(msg, act->raw_value_x2->data, act->raw_value_x2->len);
+ }
+ break;
+ }
+ default:
+ BT_ERR("Not a sensor client model message opcode");
+ err = -EINVAL;
+ goto end;
+ }
+
+ err = bt_mesh_client_send_msg(common->model, common->opcode, &common->ctx, msg,
+ timeout_handler, common->msg_timeout, need_ack,
+ common->cb, common->cb_data);
+ if (err) {
+ BT_ERR("Sensor message send failed (err %d)", err);
+ }
+
+end:
+ bt_mesh_free_buf(msg);
+
+ return err;
+}
+
+int bt_mesh_sensor_client_get_state(struct bt_mesh_common_param *common, void *get, void *status)
+{
+ bt_mesh_sensor_client_t *client = NULL;
+ u16_t length = 0;
+
+ if (!common || !common->model || !get) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_sensor_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_SENSOR_DESCRIPTOR_GET:
+ length = BT_MESH_SENSOR_DESCRIPTOR_GET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_GET:
+ length = BT_MESH_SENSOR_CADENCE_GET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_SENSOR_SETTINGS_GET:
+ length = BT_MESH_SENSOR_SETTINGS_GET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_GET:
+ length = BT_MESH_SENSOR_SETTING_GET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_SENSOR_GET:
+ length = BT_MESH_SENSOR_GET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_SENSOR_COLUMN_GET: {
+ struct bt_mesh_sensor_column_get *value;
+ value = (struct bt_mesh_sensor_column_get *)get;
+ if (!value->raw_value_x) {
+ BT_ERR("Sensor column get is NULL");
+ return -EINVAL;
+ }
+ length = (2 + 2 + value->raw_value_x->len + 4);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SERIES_GET: {
+ struct bt_mesh_sensor_series_get *value;
+ value = (struct bt_mesh_sensor_series_get *)get;
+ if (value->op_en) {
+ if (!value->raw_value_x1 || !value->raw_value_x2) {
+ BT_ERR("Sensor series get parameter is NULL");
+ return -EINVAL;
+ }
+ }
+ if (value->op_en) {
+ length = value->raw_value_x1->len + value->raw_value_x2->len;
+ }
+ length += (2 + 2 + 4);
+ break;
+ }
+ default:
+ BT_ERR("Not a sensor get message opcode");
+ return -EINVAL;
+ }
+
+ return sensor_act_state(common, get, length, true);
+}
+
+int bt_mesh_sensor_client_set_state(struct bt_mesh_common_param *common, void *set, void *status)
+{
+ bt_mesh_sensor_client_t *client = NULL;
+ u16_t length = 0;
+ bool need_ack = false;
+
+ if (!common || !common->model || !set) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_sensor_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_SENSOR_CADENCE_SET_UNACK: {
+ struct bt_mesh_sensor_cadence_set *value;
+ value = (struct bt_mesh_sensor_cadence_set *)set;
+ if (!value->status_trigger_delta_down || !value->status_trigger_delta_up ||
+ !value->fast_cadence_low || !value->fast_cadence_high) {
+ BT_ERR("Sensor cadence set parameter is NULL");
+ return -EINVAL;
+ }
+ length = value->status_trigger_delta_down->len + \
+ value->status_trigger_delta_up->len + \
+ value->fast_cadence_low->len + \
+ value->fast_cadence_high->len;
+ length += (1 + 2 + 1 + 1 + 4);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_SENSOR_SETTING_SET_UNACK: {
+ struct bt_mesh_sensor_setting_set *value;
+ value = (struct bt_mesh_sensor_setting_set *)set;
+ if (!value->sensor_setting_raw) {
+ BT_ERR("Sensor_setting_raw is NULL");
+ return -EINVAL;
+ }
+ length = (1 + 2 + 2 + value->sensor_setting_raw->len + 4);
+ break;
+ }
+ default:
+ BT_ERR("Not a sensor set message opcode");
+ return -EINVAL;
+ }
+
+ return sensor_act_state(common, set, length, need_ack);
+}
+
+int bt_mesh_sensor_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ bt_mesh_sensor_client_t *client = NULL;
+ sensor_internal_data_t *internal = NULL;
+
+ BT_DBG("primary %u", primary);
+
+ if (!model) {
+ BT_ERR("Sensor client model is NULL");
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_sensor_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("Sensor client model user_data is NULL");
+ return -EINVAL;
+ }
+
+ /* TODO: call osi_free() when deinit function is invoked*/
+ internal = osi_calloc(sizeof(sensor_internal_data_t));
+ if (!internal) {
+ BT_ERR("Allocate memory for sensor internal data fail");
+ return -ENOMEM;
+ }
+
+ sys_slist_init(&internal->queue);
+
+ client->model = model;
+ client->op_pair_size = ARRAY_SIZE(sensor_op_pair);
+ client->op_pair = sensor_op_pair;
+ client->internal_data = internal;
+
+ return 0;
+}
\ No newline at end of file
diff --git a/components/bt/ble_mesh/mesh_models/time_scene_client.c b/components/bt/ble_mesh/mesh_models/time_scene_client.c
new file mode 100644
index 0000000000..29c7fbea92
--- /dev/null
+++ b/components/bt/ble_mesh/mesh_models/time_scene_client.c
@@ -0,0 +1,696 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+#include
+
+#include "osi/allocator.h"
+
+#include "mesh_types.h"
+#include "mesh_kernel.h"
+#include "mesh_trace.h"
+#include "mesh.h"
+
+#include "model_op.h"
+#include "common.h"
+#include "time_scene_client.h"
+
+#include "sdkconfig.h"
+
+#include "btc_ble_mesh_time_scene_client.h"
+
+/** The following are the macro definitions of time and client
+ * scene model messages length, and a message is composed of
+ * three parts: Opcode + msg_value + MIC
+ */
+/* Time client messages length */
+#define BT_MESH_TIME_SET_MSG_LEN (1 + 10 + 4)
+#define BT_MESH_TIME_ZONE_SET_MSG_LEN (2 + 6 + 4)
+#define BT_MESH_TAI_UTC_DELTA_SET_MSG_LEN (2 + 7 + 4)
+#define BT_MESH_TIME_ROLE_SET_MSG_LEN (2 + 1 + 4)
+
+/* Scene client messages length */
+#define BT_MESH_SCENE_STORE_MSG_LEN (2 + 2 + 4)
+#define BT_MESH_SCENE_RECALL_MSG_LEN (2 + 5 + 4)
+#define BT_MESH_SCENE_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_SCENE_REGISTER_GET_MSG_LEN (2 + 0 + 4)
+#define BT_MESH_SCENE_DELETE_MSG_LEN (2 + 2 + 4)
+
+/* Scheduler client messages length */
+#define BT_MESH_SCHEDULER_ACT_GET_MSG_LEN (2 + 1 + 4)
+#define BT_MESH_SCHEDULER_ACT_SET_MSG_LEN (1 + 10 + 4)
+
+#define BT_MESH_SCENE_GET_STATE_MSG_LEN (2 + 1 + 4)
+#define BT_MESH_SCENE_ACT_STATE_MSG_LEN (2 + 2 + 4)
+
+static const bt_mesh_client_op_pair_t time_scene_op_pair[] = {
+ { BT_MESH_MODEL_OP_TIME_GET, BT_MESH_MODEL_OP_TIME_STATUS },
+ { BT_MESH_MODEL_OP_TIME_SET, BT_MESH_MODEL_OP_TIME_STATUS },
+ { BT_MESH_MODEL_OP_TIME_ZONE_GET, BT_MESH_MODEL_OP_TIME_ZONE_STATUS },
+ { BT_MESH_MODEL_OP_TIME_ZONE_SET, BT_MESH_MODEL_OP_TIME_ZONE_STATUS },
+ { BT_MESH_MODEL_OP_TAI_UTC_DELTA_GET, BT_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS },
+ { BT_MESH_MODEL_OP_TAI_UTC_DELTA_SET, BT_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS },
+ { BT_MESH_MODEL_OP_TIME_ROLE_GET, BT_MESH_MODEL_OP_TIME_ROLE_STATUS },
+ { BT_MESH_MODEL_OP_TIME_ROLE_SET, BT_MESH_MODEL_OP_TIME_ROLE_STATUS },
+ { BT_MESH_MODEL_OP_SCENE_STORE, BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS },
+ { BT_MESH_MODEL_OP_SCENE_RECALL, BT_MESH_MODEL_OP_SCENE_STATUS },
+ { BT_MESH_MODEL_OP_SCENE_GET, BT_MESH_MODEL_OP_SCENE_STATUS },
+ { BT_MESH_MODEL_OP_SCENE_REGISTER_GET, BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS },
+ { BT_MESH_MODEL_OP_SCENE_DELETE, BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS },
+ { BT_MESH_MODEL_OP_SCHEDULER_GET, BT_MESH_MODEL_OP_SCHEDULER_STATUS },
+ { BT_MESH_MODEL_OP_SCHEDULER_ACT_GET, BT_MESH_MODEL_OP_SCHEDULER_ACT_STATUS },
+ { BT_MESH_MODEL_OP_SCHEDULER_ACT_SET, BT_MESH_MODEL_OP_SCHEDULER_ACT_STATUS },
+};
+
+static void timeout_handler(struct k_work *work)
+{
+ bt_mesh_time_scene_client_t *client = NULL;
+ time_scene_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+
+ BT_WARN("Receive time scene status message timeout");
+
+ node = CONTAINER_OF(work, bt_mesh_client_node_t, timer.work);
+ if (!node || !node->ctx.model) {
+ BT_ERR("%s: node parameter is NULL", __func__);
+ return;
+ }
+
+ client = (bt_mesh_time_scene_client_t *)node->ctx.model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (time_scene_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: internal_data is NULL", __func__);
+ return;
+ }
+
+ bt_mesh_callback_time_scene_status_to_btc(node->opcode, 0x03, node->ctx.model,
+ &node->ctx, NULL, 0);
+
+ bt_mesh_client_free_node(&internal->queue, node);
+
+ return;
+}
+
+static void time_scene_status(struct bt_mesh_model *model,
+ struct bt_mesh_msg_ctx *ctx,
+ struct net_buf_simple *buf)
+{
+ bt_mesh_time_scene_client_t *client = NULL;
+ time_scene_internal_data_t *internal = NULL;
+ bt_mesh_client_node_t *node = NULL;
+ u8_t *val = NULL;
+ u8_t evt = 0xFF;
+ u32_t rsp = 0;
+ size_t len = 0;
+
+ BT_DBG("%s: len %d, bytes %s", __func__, buf->len, bt_hex(buf->data, buf->len));
+
+ client = (bt_mesh_time_scene_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("%s: model user_data is NULL", __func__);
+ return;
+ }
+
+ internal = (time_scene_internal_data_t *)client->internal_data;
+ if (!internal) {
+ BT_ERR("%s: model internal_data is NULL", __func__);
+ return;
+ }
+
+ rsp = ctx->recv_op;
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_TIME_STATUS: {
+ struct bt_mesh_time_status *status = NULL;
+ if (buf->len != 5 && buf->len != 10) {
+ BT_ERR("Wrong Time status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_time_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ memcpy(status->tai_seconds, buf->data, 5);
+ net_buf_simple_pull(buf, 5);
+ status->sub_second = net_buf_simple_pull_u8(buf);
+ status->uncertainty = net_buf_simple_pull_u8(buf);
+ u16_t temp = net_buf_simple_pull_le16(buf);
+ status->time_authority = temp & BIT(0);
+ status->tai_utc_delta = temp >> 15;
+ status->time_zone_offset = net_buf_simple_pull_u8(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_time_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_TIME_ZONE_STATUS: {
+ struct bt_mesh_time_zone_status *status = NULL;
+ if (buf->len != 7) {
+ BT_ERR("Wrong Time zone status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_time_zone_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->time_zone_offset_curr = net_buf_simple_pull_u8(buf);
+ status->time_zone_offset_new = net_buf_simple_pull_u8(buf);
+ memcpy(status->tai_zone_change, buf->data, 5);
+ net_buf_simple_pull(buf, 5);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_time_zone_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS: {
+ struct bt_mesh_tai_utc_delta_status *status = NULL;
+ if (buf->len != 9) {
+ BT_ERR("Wrong TAI UTC delta status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_tai_utc_delta_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ u16_t temp = net_buf_simple_pull_le16(buf);
+ status->tai_utc_delta_curr = temp & BIT_MASK(15);
+ status->padding_1 = (temp >> 15) & BIT(0);
+ temp = net_buf_simple_pull_le16(buf);
+ status->tai_utc_delta_new = temp & BIT_MASK(15);
+ status->padding_2 = (temp >> 15) & BIT(0);
+ memcpy(status->tai_delta_change, buf->data, 5);
+ net_buf_simple_pull(buf, 5);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_tai_utc_delta_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_TIME_ROLE_STATUS: {
+ struct bt_mesh_time_role_status *status = NULL;
+ if (buf->len != 1) {
+ BT_ERR("Wrong Time role status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_time_role_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->time_role = net_buf_simple_pull_u8(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_time_role_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_STATUS: {
+ struct bt_mesh_scene_status *status = NULL;
+ if (buf->len != 3 && buf->len != 6) {
+ BT_ERR("Wrong Scene status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_scene_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->status_code = net_buf_simple_pull_u8(buf);
+ status->current_scene = net_buf_simple_pull_le16(buf);
+ if (buf->len) {
+ status->op_en = true;
+ status->target_scene = net_buf_simple_pull_le16(buf);
+ status->remain_time = net_buf_simple_pull_u8(buf);
+ }
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_scene_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS: {
+ struct bt_mesh_scene_register_status *status = NULL;
+ status = osi_calloc(sizeof(struct bt_mesh_scene_register_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->status_code = net_buf_simple_pull_u8(buf);
+ status->current_scene = net_buf_simple_pull_le16(buf);
+ status->scenes = bt_mesh_alloc_buf(buf->len);
+ if (!status->scenes) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ osi_free(status);
+ return;
+ }
+ net_buf_simple_init(status->scenes, 0);
+ net_buf_simple_add_mem(status->scenes, buf->data, buf->len);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_scene_register_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCHEDULER_STATUS: {
+ struct bt_mesh_scheduler_status *status = NULL;
+ if (buf->len != 2) {
+ BT_ERR("Wrong Scheduler status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_scheduler_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ status->schedules = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_scheduler_status);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_STATUS: {
+ struct bt_mesh_scheduler_act_status *status = NULL;
+ if (buf->len != 10) {
+ BT_ERR("Wrong Scheduler action status length %d", buf->len);
+ return;
+ }
+ status = osi_calloc(sizeof(struct bt_mesh_scheduler_act_status));
+ if (!status) {
+ BT_ERR("%s: allocate memory fail", __func__);
+ return;
+ }
+ memcpy(status, buf->data, offsetof(struct bt_mesh_scheduler_act_status, scene_number));
+ net_buf_simple_pull(buf, offsetof(struct bt_mesh_scheduler_act_status, scene_number));
+ status->scene_number = net_buf_simple_pull_le16(buf);
+ val = (u8_t *)status;
+ len = sizeof(struct bt_mesh_scheduler_act_status);
+ break;
+ }
+ default:
+ BT_ERR("Not a time scene status message opcode");
+ return;
+ }
+
+ buf->data = val;
+ buf->len = len;
+ node = bt_mesh_is_model_message_publish(model, ctx, buf, true);
+ if (!node) {
+ BT_DBG("Unexpected time scene status message 0x%x", rsp);
+ } else {
+ switch (node->opcode) {
+ case BT_MESH_MODEL_OP_TIME_GET:
+ case BT_MESH_MODEL_OP_TIME_ZONE_GET:
+ case BT_MESH_MODEL_OP_TAI_UTC_DELTA_GET:
+ case BT_MESH_MODEL_OP_TIME_ROLE_GET:
+ case BT_MESH_MODEL_OP_SCENE_GET:
+ case BT_MESH_MODEL_OP_SCENE_REGISTER_GET:
+ case BT_MESH_MODEL_OP_SCHEDULER_GET:
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_GET:
+ evt = 0x00;
+ break;
+ case BT_MESH_MODEL_OP_TIME_SET:
+ case BT_MESH_MODEL_OP_TIME_ZONE_SET:
+ case BT_MESH_MODEL_OP_TAI_UTC_DELTA_SET:
+ case BT_MESH_MODEL_OP_TIME_ROLE_SET:
+ case BT_MESH_MODEL_OP_SCENE_STORE:
+ case BT_MESH_MODEL_OP_SCENE_RECALL:
+ case BT_MESH_MODEL_OP_SCENE_DELETE:
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_SET:
+ evt = 0x01;
+ break;
+ default:
+ break;
+ }
+
+ bt_mesh_callback_time_scene_status_to_btc(node->opcode, evt, model, ctx, val, len);
+ // Don't forget to release the node at the end.
+ bt_mesh_client_free_node(&internal->queue, node);
+ }
+
+ switch (rsp) {
+ case BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS: {
+ struct bt_mesh_scene_register_status *status;
+ status = (struct bt_mesh_scene_register_status *)val;
+ bt_mesh_free_buf(status->scenes);
+ break;
+ }
+ default:
+ break;
+ }
+
+ osi_free(val);
+
+ return;
+}
+
+const struct bt_mesh_model_op time_cli_op[] = {
+ { BT_MESH_MODEL_OP_TIME_STATUS, 5, time_scene_status },
+ { BT_MESH_MODEL_OP_TIME_ZONE_STATUS, 7, time_scene_status },
+ { BT_MESH_MODEL_OP_TAI_UTC_DELTA_STATUS, 9, time_scene_status },
+ { BT_MESH_MODEL_OP_TIME_ROLE_STATUS, 1, time_scene_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op scene_cli_op[] = {
+ { BT_MESH_MODEL_OP_SCENE_STATUS, 3, time_scene_status },
+ { BT_MESH_MODEL_OP_SCENE_REGISTER_STATUS, 3, time_scene_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+const struct bt_mesh_model_op scheduler_cli_op[] = {
+ { BT_MESH_MODEL_OP_SCHEDULER_STATUS, 2, time_scene_status },
+ { BT_MESH_MODEL_OP_SCHEDULER_ACT_STATUS, 10, time_scene_status },
+ BT_MESH_MODEL_OP_END,
+};
+
+static int time_scene_get_state(struct bt_mesh_common_param *common, void *value)
+{
+ struct net_buf_simple *msg = NET_BUF_SIMPLE(BT_MESH_SCENE_GET_STATE_MSG_LEN);
+ int err;
+
+ bt_mesh_model_msg_init(msg, common->opcode);
+
+ if (value) {
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_GET: {
+ struct bt_mesh_scheduler_act_get *get;
+ get = (struct bt_mesh_scheduler_act_get *)value;
+ net_buf_simple_add_u8(msg, get->index);
+ break;
+ }
+ default:
+ BT_DBG("This time scene message should be sent with NULL get pointer");
+ break;
+ }
+ }
+
+ err = bt_mesh_client_send_msg(common->model, common->opcode, &common->ctx, msg,
+ timeout_handler, common->msg_timeout, true,
+ common->cb, common->cb_data);
+ if (err) {
+ BT_ERR("Time scene get message send failed (err %d)", err);
+ }
+
+ return err;
+}
+
+static int time_scene_set_state(struct bt_mesh_common_param *common,
+ void *value, u16_t value_len, bool need_ack)
+{
+ struct net_buf_simple *msg = NULL;
+ int err;
+
+ msg = bt_mesh_alloc_buf(value_len);
+ if (!msg) {
+ BT_ERR("Time scene allocate memory fail");
+ return -ENOMEM;
+ }
+
+ bt_mesh_model_msg_init(msg, common->opcode);
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_TIME_SET: {
+ struct bt_mesh_time_set *set;
+ set = (struct bt_mesh_time_set *)value;
+ net_buf_simple_add_mem(msg, set->tai_seconds, 5);
+ net_buf_simple_add_u8(msg, set->sub_second);
+ net_buf_simple_add_u8(msg, set->uncertainty);
+ net_buf_simple_add_le16(msg, set->tai_utc_delta << 1 | set->time_authority);
+ net_buf_simple_add_u8(msg, set->time_zone_offset);
+ break;
+ }
+ case BT_MESH_MODEL_OP_TIME_ZONE_SET: {
+ struct bt_mesh_time_zone_set *set;
+ set = (struct bt_mesh_time_zone_set *)value;
+ net_buf_simple_add_u8(msg, set->time_zone_offset_new);
+ net_buf_simple_add_mem(msg, set->tai_zone_change, 5);
+ break;
+ }
+ case BT_MESH_MODEL_OP_TAI_UTC_DELTA_SET: {
+ struct bt_mesh_tai_utc_delta_set *set;
+ set = (struct bt_mesh_tai_utc_delta_set *)value;
+ net_buf_simple_add_le16(msg, set->padding << 15 | set->tai_utc_delta_new);
+ net_buf_simple_add_mem(msg, set->tai_delta_change, 5);
+ break;
+ }
+ case BT_MESH_MODEL_OP_TIME_ROLE_SET: {
+ struct bt_mesh_time_role_set *set;
+ set = (struct bt_mesh_time_role_set *)value;
+ net_buf_simple_add_u8(msg, set->time_role);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_STORE:
+ case BT_MESH_MODEL_OP_SCENE_STORE_UNACK: {
+ struct bt_mesh_scene_store *set;
+ set = (struct bt_mesh_scene_store *)value;
+ net_buf_simple_add_le16(msg, set->scene_number);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_RECALL:
+ case BT_MESH_MODEL_OP_SCENE_RECALL_UNACK: {
+ struct bt_mesh_scene_recall *set;
+ set = (struct bt_mesh_scene_recall *)value;
+ net_buf_simple_add_le16(msg, set->scene_number);
+ net_buf_simple_add_u8(msg, set->tid);
+ if (set->op_en) {
+ net_buf_simple_add_u8(msg, set->trans_time);
+ net_buf_simple_add_u8(msg, set->delay);
+ }
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_DELETE:
+ case BT_MESH_MODEL_OP_SCENE_DELETE_UNACK: {
+ struct bt_mesh_scene_delete *set;
+ set = (struct bt_mesh_scene_delete *)value;
+ net_buf_simple_add_le16(msg, set->scene_number);
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_SET:
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_SET_UNACK: {
+ struct bt_mesh_scheduler_act_set *set;
+ set = (struct bt_mesh_scheduler_act_set *)value;
+ net_buf_simple_add_mem(msg, set, offsetof(struct bt_mesh_scheduler_act_set, scene_number));
+ net_buf_simple_add_le16(msg, set->scene_number);
+ break;
+ }
+ default:
+ BT_ERR("Not a generic client model set message opcode");
+ err = -EINVAL;
+ goto end;
+ }
+
+ err = bt_mesh_client_send_msg(common->model, common->opcode, &common->ctx, msg,
+ timeout_handler, common->msg_timeout, need_ack,
+ common->cb, common->cb_data);
+ if (err) {
+ BT_ERR("Time scene set message send failed (err %d)", err);
+ }
+
+end:
+ bt_mesh_free_buf(msg);
+ return err;
+}
+
+int bt_mesh_time_scene_client_get_state(struct bt_mesh_common_param *common, void *get, void *status)
+{
+ bt_mesh_time_scene_client_t *client = NULL;
+
+ if (!common || !common->model) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_time_scene_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_TIME_GET:
+ case BT_MESH_MODEL_OP_TIME_ZONE_GET:
+ case BT_MESH_MODEL_OP_TAI_UTC_DELTA_GET:
+ case BT_MESH_MODEL_OP_TIME_ROLE_GET:
+ case BT_MESH_MODEL_OP_SCENE_GET:
+ case BT_MESH_MODEL_OP_SCENE_REGISTER_GET:
+ case BT_MESH_MODEL_OP_SCHEDULER_GET:
+ break;
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_GET:
+ if (!get) {
+ BT_ERR("Scheduler action index is NULL");
+ return -EINVAL;
+ }
+ break;
+ default:
+ BT_ERR("Not a time scene get message opcode");
+ return -EINVAL;
+ }
+
+ return time_scene_get_state(common, get);
+}
+
+int bt_mesh_time_scene_client_set_state(struct bt_mesh_common_param *common, void *set, void *status)
+{
+ bt_mesh_time_scene_client_t *client = NULL;
+ u16_t length = 0;
+ bool need_ack = false;
+
+ if (!common || !common->model || !set) {
+ BT_ERR("%s: common parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_time_scene_client_t *)common->model->user_data;
+ if (!client || !client->internal_data) {
+ BT_ERR("%s: client parameter is NULL", __func__);
+ return -EINVAL;
+ }
+
+ switch (common->opcode) {
+ case BT_MESH_MODEL_OP_TIME_SET:
+ need_ack = true;
+ length = BT_MESH_TIME_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_TIME_ZONE_SET:
+ need_ack = true;
+ length = BT_MESH_TIME_ZONE_SET_MSG_LEN;
+ break;
+ case BT_MESH_MODEL_OP_TAI_UTC_DELTA_SET: {
+ struct bt_mesh_tai_utc_delta_set *value;
+ value = (struct bt_mesh_tai_utc_delta_set *)set;
+ if (value->padding) {
+ BT_ERR("Non-zero padding value is prohibited");
+ return -EINVAL;
+ }
+ need_ack = true;
+ length = BT_MESH_TAI_UTC_DELTA_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_TIME_ROLE_SET: {
+ struct bt_mesh_time_role_set *value;
+ value = (struct bt_mesh_time_role_set *)set;
+ if (value->time_role > 0x03) {
+ BT_ERR("Time role value is prohibited");
+ return -EINVAL;
+ }
+ need_ack = true;
+ length = BT_MESH_TIME_ROLE_SET_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_STORE:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_SCENE_STORE_UNACK: {
+ struct bt_mesh_scene_store *value;
+ value = (struct bt_mesh_scene_store *)set;
+ if (!value->scene_number) {
+ BT_ERR("Scene store scene_number 0x0000 is prohibited");
+ return -EINVAL;
+ }
+ length = BT_MESH_SCENE_STORE_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_RECALL:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_SCENE_RECALL_UNACK: {
+ struct bt_mesh_scene_recall *value;
+ value = (struct bt_mesh_scene_recall *)set;
+ if (!value->scene_number) {
+ BT_ERR("Scene recall scene_number 0x0000 is prohibited");
+ return -EINVAL;
+ }
+ if (value->op_en) {
+ if ((value->trans_time & 0x3F) > 0x3E) {
+ BT_ERR("Scene recall transition time is bigger than 0x3E");
+ return -EINVAL;
+ }
+ }
+ length = BT_MESH_SCENE_RECALL_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCENE_DELETE:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_SCENE_DELETE_UNACK: {
+ length = BT_MESH_SCENE_DELETE_MSG_LEN;
+ break;
+ }
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_SET:
+ need_ack = true;
+ case BT_MESH_MODEL_OP_SCHEDULER_ACT_SET_UNACK: {
+ struct bt_mesh_scheduler_act_set *value;
+ value = (struct bt_mesh_scheduler_act_set *)set;
+ if (value->year > 0x64) {
+ BT_ERR("Scheduler register year value is prohibited");
+ return -EINVAL;
+ }
+ if (value->hour > 0x19) {
+ BT_ERR("Scheduler register hour value is prohibited");
+ return -EINVAL;
+ }
+ length = BT_MESH_SCHEDULER_ACT_SET_MSG_LEN;
+ break;
+ }
+ default:
+ BT_ERR("Not a time scene set message opcode");
+ return -EINVAL;
+ }
+
+ return time_scene_set_state(common, set, length, need_ack);
+}
+
+static int time_scene_client_init(struct bt_mesh_model *model, bool primary)
+{
+ bt_mesh_time_scene_client_t *client = NULL;
+ time_scene_internal_data_t *internal = NULL;
+
+ BT_DBG("primary %u", primary);
+
+ if (!model) {
+ BT_ERR("Time scene client model is NULL");
+ return -EINVAL;
+ }
+
+ client = (bt_mesh_time_scene_client_t *)model->user_data;
+ if (!client) {
+ BT_ERR("Time scene client model user_data is NULL");
+ return -EINVAL;
+ }
+
+ /* TODO: call osi_free() when deinit function is invoked*/
+ internal = osi_calloc(sizeof(time_scene_internal_data_t));
+ if (!internal) {
+ BT_ERR("Allocate memory for time scene internal data fail");
+ return -ENOMEM;
+ }
+
+ sys_slist_init(&internal->queue);
+
+ client->model = model;
+ client->op_pair_size = ARRAY_SIZE(time_scene_op_pair);
+ client->op_pair = time_scene_op_pair;
+ client->internal_data = internal;
+
+ return 0;
+}
+
+int bt_mesh_time_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return time_scene_client_init(model, primary);
+}
+
+int bt_mesh_scene_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return time_scene_client_init(model, primary);
+}
+
+int bt_mesh_scheduler_cli_init(struct bt_mesh_model *model, bool primary)
+{
+ return time_scene_client_init(model, primary);
+}
\ No newline at end of file
diff --git a/components/bt/bluedroid/btc/core/btc_task.c b/components/bt/bluedroid/btc/core/btc_task.c
index 1022e770d1..97c5436d06 100644
--- a/components/bt/bluedroid/btc/core/btc_task.c
+++ b/components/bt/bluedroid/btc/core/btc_task.c
@@ -46,7 +46,15 @@
#include "btc_hf_client.h"
#endif /* #if BTC_HF_CLIENT_INCLUDED */
#endif /* #if CONFIG_CLASSIC_BT_ENABLED */
-
+#if CONFIG_BT_MESH
+#include "btc_ble_mesh_prov.h"
+#include "btc_ble_mesh_health.h"
+#include "btc_ble_mesh_config_client.h"
+#include "btc_ble_mesh_generic_client.h"
+#include "btc_ble_mesh_light_client.h"
+#include "btc_ble_mesh_sensor_client.h"
+#include "btc_ble_mesh_time_scene_client.h"
+#endif /* CONFIG_BT_MESH */
static xTaskHandle xBtcTaskHandle = NULL;
static xQueueHandle xBtcQueue = 0;
@@ -87,6 +95,19 @@ static btc_func_t profile_tab[BTC_PID_NUM] = {
[BTC_PID_HF_CLIENT] = {btc_hf_client_call_handler, btc_hf_client_cb_handler},
#endif /* #if BTC_HF_CLIENT_INCLUDED */
#endif /* #if CONFIG_CLASSIC_BT_ENABLED */
+
+#if CONFIG_BT_MESH
+ [BTC_PID_PROV] = {btc_mesh_prov_call_handler, btc_mesh_prov_cb_handler},
+ [BTC_PID_MODEL] = {btc_mesh_model_call_handler, btc_mesh_model_cb_handler},
+ [BTC_PID_HEALTH_CLIENT] = {btc_mesh_health_client_call_handler, btc_mesh_health_client_cb_handler},
+ [BTC_PID_HEALTH_SERVER] = {btc_mesh_health_server_call_handler, btc_mesh_health_server_cb_handler},
+ [BTC_PID_CFG_CLIENT] = {btc_mesh_cfg_client_call_handler, btc_mesh_cfg_client_cb_handler},
+ [BTC_PID_CFG_SERVER] = {NULL , btc_mesh_cfg_server_cb_handler},
+ [BTC_PID_GENERIC_CLIENT] = {btc_mesh_generic_client_call_handler, btc_mesh_generic_client_cb_handler},
+ [BTC_PID_LIGHT_CLIENT] = {btc_mesh_light_client_call_handler, btc_mesh_light_client_cb_handler},
+ [BTC_PID_SENSOR_CLIENT] = {btc_mesh_sensor_client_call_handler, btc_mesh_sensor_client_cb_handler},
+ [BTC_PID_TIME_SCENE_CLIENT] = {btc_mesh_time_scene_client_call_handler, btc_mesh_time_scene_client_cb_handler},
+#endif /* #if CONFIG_BT_MESH */
};
/*****************************************************************************
diff --git a/components/bt/bluedroid/btc/include/btc/btc_task.h b/components/bt/bluedroid/btc/include/btc/btc_task.h
index 5813c52178..20c507cea3 100644
--- a/components/bt/bluedroid/btc/include/btc/btc_task.h
+++ b/components/bt/bluedroid/btc/include/btc/btc_task.h
@@ -63,6 +63,18 @@ typedef enum {
BTC_PID_HF_CLIENT,
#endif /* BTC_HF_CLIENT_INCLUDED */
#endif /* CONFIG_CLASSIC_BT_ENABLED */
+#if CONFIG_BT_MESH
+ BTC_PID_PROV,
+ BTC_PID_MODEL,
+ BTC_PID_HEALTH_CLIENT,
+ BTC_PID_HEALTH_SERVER,
+ BTC_PID_CFG_CLIENT,
+ BTC_PID_CFG_SERVER,
+ BTC_PID_GENERIC_CLIENT,
+ BTC_PID_LIGHT_CLIENT,
+ BTC_PID_SENSOR_CLIENT,
+ BTC_PID_TIME_SCENE_CLIENT,
+#endif /* #if CONFIG_BT_MESH */
BTC_PID_NUM,
} btc_pid_t; //btc profile id
diff --git a/components/bt/component.mk b/components/bt/component.mk
index eb9b39c02d..e6a8345f84 100644
--- a/components/bt/component.mk
+++ b/components/bt/component.mk
@@ -41,7 +41,6 @@ COMPONENT_PRIV_INCLUDEDIRS += bluedroid/bta/include \
bluedroid/device/include \
bluedroid/gki/include \
bluedroid/hci/include \
- bluedroid/osi/include \
bluedroid/utils/include \
bluedroid/external/sbc/decoder/include \
bluedroid/external/sbc/encoder/include \
@@ -71,7 +70,19 @@ COMPONENT_PRIV_INCLUDEDIRS += bluedroid/bta/include \
bluedroid/utils/include \
bluedroid/common/include
-COMPONENT_ADD_INCLUDEDIRS += bluedroid/api/include/api
+COMPONENT_ADD_INCLUDEDIRS += bluedroid/api/include/api \
+ bluedroid/osi/include \
+
+ifdef CONFIG_BT_MESH
+ COMPONENT_ADD_INCLUDEDIRS += ble_mesh/mesh_core \
+ ble_mesh/mesh_core/include \
+ ble_mesh/mesh_core/settings/include \
+ ble_mesh/btc/include \
+ ble_mesh/mesh_models/include \
+ ble_mesh/api/core/include \
+ ble_mesh/api/models/include \
+ ble_mesh/api
+endif
COMPONENT_SRCDIRS += bluedroid/bta/dm \
bluedroid/bta/gatt \
@@ -120,6 +131,15 @@ COMPONENT_SRCDIRS += bluedroid/bta/dm \
bluedroid/api \
bluedroid
+ifdef CONFIG_BT_MESH
+ COMPONENT_SRCDIRS += ble_mesh/mesh_core \
+ ble_mesh/mesh_core/settings \
+ ble_mesh/btc \
+ ble_mesh/mesh_models \
+ ble_mesh/api/core \
+ ble_mesh/api/models
+endif
+
ifeq ($(GCC_NOT_5_2_0), 1)
bluedroid/bta/sdp/bta_sdp_act.o: CFLAGS += -Wno-unused-const-variable
bluedroid/btc/core/btc_config.o: CFLAGS += -Wno-unused-const-variable
diff --git a/components/bt/test/test_ble_mesh_prov.c b/components/bt/test/test_ble_mesh_prov.c
new file mode 100644
index 0000000000..9db16a097c
--- /dev/null
+++ b/components/bt/test/test_ble_mesh_prov.c
@@ -0,0 +1,280 @@
+/*
+ Tests for the BLE Mesh Client API and timer implementation
+*/
+
+#include
+#include
+#include
+#include
+#include
+#include
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "freertos/semphr.h"
+#include "freertos/queue.h"
+#include "freertos/xtensa_api.h"
+#include "unity.h"
+#include "esp_heap_caps.h"
+#include "esp_log.h"
+#include "freertos/ringbuf.h"
+#include "esp_system.h"
+#include "nvs_flash.h"
+
+#include "esp_bt.h"
+#include "esp_bt_main.h"
+#include "esp_bt_device.h"
+#include "esp_gap_ble_api.h"
+#include "esp_ble_mesh_provisioning_api.h"
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_config_api.h"
+
+#define TAG "ble_mesh_test"
+
+static SemaphoreHandle_t ble_mesh_test_sem;
+static int error_code = -1;
+
+static esp_ble_mesh_model_pub_t gen_onoff_pub = {
+ .msg = NET_BUF_SIMPLE(2 + 1),
+ .update = NULL,
+};
+
+static esp_ble_mesh_model_op_t gen_onoff_op[] = {
+ { ESP_BLE_MESH_MODEL_OP_2(0x82, 0x01), 0, 0},
+ { ESP_BLE_MESH_MODEL_OP_2(0x82, 0x02), 2, 0},
+ { ESP_BLE_MESH_MODEL_OP_2(0x82, 0x03), 2, 0},
+ /* Each model operation struct array must use this terminator
+ * as the end tag of the operation uint. */
+ ESP_BLE_MESH_MODEL_OP_END,
+ };
+
+/* Define the SIG light onoff model */
+static esp_ble_mesh_model_t root_models[] = {
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV, gen_onoff_op,
+ &gen_onoff_pub, NULL),
+};
+
+/* Define the vendor light model operation */
+static esp_ble_mesh_model_op_t light_op[] = {
+ {ESP_BLE_MESH_MODEL_OP_3(0x01, 0x0D01), 1, 0}, //BRIGHINESS_GET
+ {ESP_BLE_MESH_MODEL_OP_3(0x10, 0x0D10), 1, 0}, //BRIGHINESS_GET_RESULT
+ {ESP_BLE_MESH_MODEL_OP_3(0x08, 0x0D08), 1, 0}, //BRIGHINESS_SET
+ /* Each model operation struct array must use this terminator
+ * as the end tag of the operation uint. */
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+/* Define the vendor light model, the componey id is 0x01, and the model id is 0x7802 */
+static esp_ble_mesh_model_t light_model[] = {
+ ESP_BLE_MESH_VENDOR_MODEL(0x01, 0x7802, light_op, NULL, NULL),
+};
+
+/* There are two element will register to the stack, one is the sig onoff model, the
+ other is vendor light model */
+static esp_ble_mesh_elem_t elements[] = {
+ ESP_BLE_MESH_ELEM(0, root_models, ESP_BLE_MESH_MODEL_NONE),
+ ESP_BLE_MESH_ELEM(0, ESP_BLE_MESH_MODEL_NONE, light_model),
+};
+
+static const esp_ble_mesh_comp_t comp = {
+ .cid = 0,
+ .elem = elements,
+ .elem_count = ARRAY_SIZE(elements),
+};
+
+static uint8_t dev_uuid[16] = { 0xdd, 0xdd };
+
+/* Disable OOB security for SILabs Android app */
+static esp_ble_mesh_prov_t prov = {
+ .uuid = dev_uuid,
+#if 0
+ .output_size = 4,
+ .output_actions = ESP_BLE_MESH_DISPLAY_NUMBER,
+ .input_actions = ESP_BLE_MESH_PUSH,
+ .input_size = 4,
+#else
+ .output_size = 0,
+ .output_actions = 0,
+#endif
+};
+
+static char *esp_ble_mesh_prov_event_to_str(esp_ble_mesh_prov_cb_event_t event)
+{
+ switch(event) {
+ case ESP_BLE_MESH_PROV_REGISTER_COMP_EVT:
+ return "ESP_BLE_MESH_PROV_REGISTER_COMP_EVT";
+ case ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT";
+ case ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT";
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT";
+ case ESP_BLE_MESH_NODE_PROV_INPUT_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_INPUT_EVT";
+ case ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT";
+ case ESP_BLE_MESH_NODE_PROV_RESET_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_RESET_EVT";
+ //case ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT:
+ // return "ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT";
+ //case ESP_BLE_MESH_MODEL_OPERATION_EVT:
+ // return "ESP_BLE_MESH_MODEL_OPERATION_EVT";
+ default:
+ return "Invalid mesh provision event.";
+ }
+
+ return NULL;
+}
+
+static void esp_ble_mesh_prov_cb(esp_ble_mesh_prov_cb_event_t event,
+ esp_ble_mesh_prov_cb_param_t *param)
+{
+ ESP_LOGE(TAG, "%s, event = %s", __func__, esp_ble_mesh_prov_event_to_str(event));
+ switch(event) {
+ case ESP_BLE_MESH_PROVISIONER_DEV_ADD_COMP_EVT:
+ error_code = param->provisioner_dev_add_comp.err_code;
+ xSemaphoreGive(ble_mesh_test_sem);
+ break;
+ case ESP_BLE_MESH_PROVISIONER_DEV_DEL_COMP_EVT:
+ error_code = param->provisioner_dev_del_comp.err_code;
+ xSemaphoreGive(ble_mesh_test_sem);
+ break;
+ case ESP_BLE_MESH_PROVISIONER_SET_NODE_NAME_COMP_EVT:
+ error_code = param->set_node_name_comp.err_code;
+ xSemaphoreGive(ble_mesh_test_sem);
+ break;
+ case ESP_BLE_MESH_PROVISIONER_ADD_LOCAL_APP_KEY_COMP_EVT:
+ error_code = param->add_local_app_key_comp.err_code;
+ xSemaphoreGive(ble_mesh_test_sem);
+ break;
+ default:
+ break;
+ }
+ return;
+}
+
+static int ble_mesh_init(void)
+{
+ int err = 0;
+
+ // Register the provisioning callback function
+ esp_ble_mesh_register_prov_callback(esp_ble_mesh_prov_cb);
+ memcpy(dev_uuid, esp_bt_dev_get_address(), 6);
+
+ err = esp_ble_mesh_init(&prov, &comp);
+ if (err) {
+ ESP_LOGE(TAG, "Initializing mesh failed (err %d)\n", err);
+ return err;
+ }
+
+ esp_ble_mesh_node_prov_enable(BT_MESH_PROV_ADV | BT_MESH_PROV_GATT);
+
+ ESP_LOGI(TAG, "Mesh initialized\n");
+
+ return err;
+}
+
+static esp_err_t bluetooth_init(void)
+{
+ esp_err_t ret;
+
+ ret = nvs_flash_init();
+ if (ret == ESP_ERR_NVS_NO_FREE_PAGES) {
+ ESP_ERROR_CHECK(nvs_flash_erase());
+ ret = nvs_flash_init();
+ }
+ ESP_ERROR_CHECK(ret);
+
+ ESP_ERROR_CHECK(esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT));
+
+ esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
+ ret = esp_bt_controller_init(&bt_cfg);
+ if (ret) {
+ ESP_LOGE(TAG, "%s initialize controller failed\n", __func__);
+ return ret;
+ }
+
+ ret = esp_bt_controller_enable(ESP_BT_MODE_BLE);
+ if (ret) {
+ ESP_LOGE(TAG, "%s enable controller failed\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_init();
+ if (ret) {
+ ESP_LOGE(TAG, "%s init bluetooth failed\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_enable();
+ if (ret) {
+ ESP_LOGE(TAG, "%s enable bluetooth failed\n", __func__);
+ return ret;
+ }
+
+ return ret;
+}
+
+static esp_err_t test_main(void)
+{
+ int err;
+
+ ESP_LOGI(TAG, "Initializing...");
+
+ err = bluetooth_init();
+ if (err) {
+ ESP_LOGE(TAG, "esp32_bluetooth_init failed (err %d)", err);
+ return err;
+ }
+
+ //esp_ble_gap_config_local_privacy(true);
+
+ /* Initialize the Bluetooth Mesh Subsystem */
+ err = ble_mesh_init();
+ if (err) {
+ ESP_LOGE(TAG, "Bluetooth mesh init failed (err %d)\n", err);
+ return err;
+ }
+
+ return 0;
+}
+
+extern const struct bt_mesh_comp *dev_comp;
+
+TEST_CASE("ble_mesh_provisioner_cb", "[ble_mesh]")
+{
+ esp_bd_addr_t addr;
+ esp_err_t ret = test_main();
+ ble_mesh_test_sem = xSemaphoreCreateCounting(1, 0); //Max 1, initial 0
+ memcpy(dev_uuid, esp_bt_dev_get_address(), ESP_BD_ADDR_LEN);
+ memcpy(addr, esp_bt_dev_get_address(), ESP_BD_ADDR_LEN);
+ // Test the device add api
+ TEST_ASSERT((esp_ble_mesh_provisioner_add_unprov_dev(addr, BLE_ADDR_TYPE_PUBLIC, dev_uuid,
+ ESP_BLE_MESH_PROV_ADV | ESP_BLE_MESH_PROV_GATT, false)) == ESP_OK);
+ xSemaphoreTake(ble_mesh_test_sem, portMAX_DELAY);
+ // Should callback 0 when device add complete
+ TEST_ASSERT(error_code == 0);
+ // Test the device delete api
+ TEST_ASSERT((esp_ble_mesh_provisioner_delete_dev(addr, BLE_ADDR_TYPE_PUBLIC, dev_uuid)) == ESP_OK);
+ xSemaphoreTake(ble_mesh_test_sem, portMAX_DELAY);
+ // Should callback 0 when device delete complete
+ TEST_ASSERT(error_code == 0);
+ char *node_name = "esp_ble_mesh";
+ TEST_ASSERT((esp_ble_mesh_provisioner_set_node_name(0, node_name)) == ESP_OK);
+ xSemaphoreTake(ble_mesh_test_sem, portMAX_DELAY);
+ // Should callback 0 when set node name complete
+ TEST_ASSERT(error_code == 0);
+ // Get the node name after set
+ char *node_name_get = esp_ble_mesh_provisioner_get_node_name(0);
+ // The node name should be equal with that set at the begining.
+ TEST_ASSERT(strcmp(node_name_get, "esp_ble_mesh") == 0);
+ int index = esp_ble_mesh_provisioner_get_node_index("esp_ble_mesh");
+ TEST_ASSERT(index == 0);
+ const uint8_t app_key[16] = {0x55};
+ uint16_t net_idx = 0;
+ uint16_t app_idx = 0;
+ TEST_ASSERT((esp_ble_mesh_provisioner_add_local_app_key(app_key, net_idx, app_idx)) == ESP_OK);
+ xSemaphoreTake(ble_mesh_test_sem, portMAX_DELAY);
+ // Should callback 0 when local app key add complete
+ TEST_ASSERT(error_code == 0);
+
+}
+
diff --git a/components/bt/test/test_client_api.c b/components/bt/test/test_client_api.c
new file mode 100644
index 0000000000..10b3fbce38
--- /dev/null
+++ b/components/bt/test/test_client_api.c
@@ -0,0 +1,313 @@
+/*
+ Tests for the BLE Mesh Client API and timer implementation
+*/
+
+#include
+#include
+#include
+#include
+#include
+#include
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "freertos/semphr.h"
+#include "freertos/queue.h"
+#include "freertos/xtensa_api.h"
+#include "unity.h"
+#include "esp_heap_caps.h"
+#include "esp_log.h"
+#include "freertos/ringbuf.h"
+#include "esp_system.h"
+#include "nvs_flash.h"
+
+#include "access.h"
+
+#include "esp_bt.h"
+#include "esp_bt_main.h"
+#include "esp_bt_device.h"
+#include "esp_gap_ble_api.h"
+#include "esp_ble_mesh_provisioning_api.h"
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_config_api.h"
+
+#define TAG "ble_mesh_test"
+
+static esp_ble_mesh_model_pub_t gen_onoff_pub = {
+ .msg = NET_BUF_SIMPLE(2 + 1),
+ .update = NULL,
+};
+
+static esp_ble_mesh_model_op_t gen_onoff_op[] = {
+ { ESP_BLE_MESH_MODEL_OP_2(0x82, 0x01), 0, 0},
+ { ESP_BLE_MESH_MODEL_OP_2(0x82, 0x02), 2, 0},
+ { ESP_BLE_MESH_MODEL_OP_2(0x82, 0x03), 2, 0},
+ /* Each model operation struct array must use this terminator
+ * as the end tag of the operation uint. */
+ ESP_BLE_MESH_MODEL_OP_END,
+ };
+
+/* Define the SIG light onoff model */
+static esp_ble_mesh_model_t root_models[] = {
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV, gen_onoff_op,
+ &gen_onoff_pub, NULL),
+};
+
+/* Define the vendor light model operation */
+static esp_ble_mesh_model_op_t light_op[] = {
+ {ESP_BLE_MESH_MODEL_OP_3(0x01, 0x0D01), 1, 0}, //BRIGHINESS_GET
+ {ESP_BLE_MESH_MODEL_OP_3(0x10, 0x0D10), 1, 0}, //BRIGHINESS_GET_RESULT
+ {ESP_BLE_MESH_MODEL_OP_3(0x08, 0x0D08), 1, 0}, //BRIGHINESS_SET
+ /* Each model operation struct array must use this terminator
+ * as the end tag of the operation uint. */
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+/* Define the vendor light model, the componey id is 0x01, and the model id is 0x7802 */
+static esp_ble_mesh_model_t light_model[] = {
+ ESP_BLE_MESH_VENDOR_MODEL(0x01, 0x7802, light_op, NULL, NULL),
+};
+
+/* There are two element will register to the stack, one is the sig onoff model, the
+ other is vendor light model */
+static esp_ble_mesh_elem_t elements[] = {
+ ESP_BLE_MESH_ELEM(0, root_models, ESP_BLE_MESH_MODEL_NONE),
+ ESP_BLE_MESH_ELEM(0, ESP_BLE_MESH_MODEL_NONE, light_model),
+};
+
+static const esp_ble_mesh_comp_t comp = {
+ .cid = 0,
+ .elem = elements,
+ .elem_count = ARRAY_SIZE(elements),
+};
+
+static uint8_t dev_uuid[16] = { 0xdd, 0xdd };
+
+/* Disable OOB security for SILabs Android app */
+static esp_ble_mesh_prov_t prov = {
+ .uuid = dev_uuid,
+#if 0
+ .output_size = 4,
+ .output_actions = ESP_BLE_MESH_DISPLAY_NUMBER,
+ .input_actions = ESP_BLE_MESH_PUSH,
+ .input_size = 4,
+#else
+ .output_size = 0,
+ .output_actions = 0,
+#endif
+};
+
+static char *esp_ble_mesh_prov_event_to_str(esp_ble_mesh_prov_cb_event_t event)
+{
+ switch(event) {
+ case ESP_BLE_MESH_PROV_REGISTER_COMP_EVT:
+ return "ESP_BLE_MESH_PROV_REGISTER_COMP_EVT";
+ case ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT";
+ case ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT";
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT";
+ case ESP_BLE_MESH_NODE_PROV_INPUT_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_INPUT_EVT";
+ case ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT";
+ case ESP_BLE_MESH_NODE_PROV_RESET_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_RESET_EVT";
+ //case ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT:
+ // return "ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT";
+ //case ESP_BLE_MESH_MODEL_OPERATION_EVT:
+ // return "ESP_BLE_MESH_MODEL_OPERATION_EVT";
+ default:
+ return "Invalid mesh provision event.";
+ }
+
+ return NULL;
+}
+
+static void esp_ble_mesh_prov_cb(esp_ble_mesh_prov_cb_event_t event,
+ esp_ble_mesh_prov_cb_param_t *param)
+{
+ ESP_LOGE(TAG, "%s, event = %s", __func__, esp_ble_mesh_prov_event_to_str(event));
+ switch(event) {
+ case ESP_BLE_MESH_PROV_REGISTER_COMP_EVT:
+ break;
+ case ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT:
+ break;
+ case ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT:
+ break;
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT:
+
+ break;
+ case ESP_BLE_MESH_NODE_PROV_INPUT_EVT:
+ break;
+ case ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT:
+ //prov_complete(param->prov_comp.net_idx,param->prov_comp.addr);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_RESET_EVT:
+ break;
+ default:
+ break;
+ }
+ return;
+}
+
+void esp_ble_mesh_model_cb(esp_ble_mesh_model_cb_event_t event,
+ esp_ble_mesh_model_cb_param_t *param)
+{
+ switch(event) {
+ case ESP_BLE_MESH_MODEL_OPERATION_EVT:
+ break;
+ case ESP_BLE_MESH_SERVER_MODEL_SEND_COMP_EVT:
+ break;
+ case ESP_BLE_MESH_MODEL_PUBLISH_COMP_EVT:
+ break;
+ default:
+ break;
+ }
+}
+
+static int ble_mesh_init(void)
+{
+ int err = 0;
+
+ // Register the provisioning callback function
+ esp_ble_mesh_register_prov_callback(esp_ble_mesh_prov_cb);
+ // Register the model send callback function
+ esp_ble_mesh_register_custom_model_callback(esp_ble_mesh_model_cb);
+ memcpy(dev_uuid, esp_bt_dev_get_address(), 6);
+
+ err = esp_ble_mesh_init(&prov, &comp);
+ if (err) {
+ ESP_LOGE(TAG, "Initializing mesh failed (err %d)\n", err);
+ return err;
+ }
+
+ esp_ble_mesh_node_prov_enable(BT_MESH_PROV_ADV | BT_MESH_PROV_GATT);
+
+ ESP_LOGI(TAG, "Mesh initialized\n");
+
+ return err;
+}
+
+static esp_err_t bluetooth_init(void)
+{
+ esp_err_t ret;
+
+ ret = nvs_flash_init();
+ if (ret == ESP_ERR_NVS_NO_FREE_PAGES) {
+ ESP_ERROR_CHECK(nvs_flash_erase());
+ ret = nvs_flash_init();
+ }
+ ESP_ERROR_CHECK(ret);
+
+ ESP_ERROR_CHECK(esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT));
+
+ esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
+ ret = esp_bt_controller_init(&bt_cfg);
+ if (ret) {
+ ESP_LOGE(TAG, "%s initialize controller failed\n", __func__);
+ return ret;
+ }
+
+ ret = esp_bt_controller_enable(ESP_BT_MODE_BLE);
+ if (ret) {
+ ESP_LOGE(TAG, "%s enable controller failed\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_init();
+ if (ret) {
+ ESP_LOGE(TAG, "%s init bluetooth failed\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_enable();
+ if (ret) {
+ ESP_LOGE(TAG, "%s enable bluetooth failed\n", __func__);
+ return ret;
+ }
+
+ return ret;
+}
+
+static esp_err_t test_main(void)
+{
+ int err;
+
+ ESP_LOGI(TAG, "Initializing...");
+
+ err = bluetooth_init();
+ if (err) {
+ ESP_LOGE(TAG, "esp32_bluetooth_init failed (err %d)", err);
+ return err;
+ }
+
+ //esp_ble_gap_config_local_privacy(true);
+
+ /* Initialize the Bluetooth Mesh Subsystem */
+ err = ble_mesh_init();
+ if (err) {
+ ESP_LOGE(TAG, "Bluetooth mesh init failed (err %d)\n", err);
+ return err;
+ }
+
+ return 0;
+}
+
+TEST_CASE("ble_mesh_node_init", "[ble_mesh]")
+{
+ esp_err_t ret = test_main();
+ TEST_ASSERT(ret == ESP_OK);
+ /* We wait init finish 200ms here */
+ vTaskDelay(200 / portTICK_PERIOD_MS);
+ const struct bt_mesh_comp *dev_comp = bt_mesh_comp_get();
+ /* We need to check if the corret model and element values are stored in
+ the dev_comp pointer. */
+ TEST_ASSERT(dev_comp);
+ /* We are register the cid = 0 to the stack, so we should check if its value is correct */
+ TEST_ASSERT(dev_comp->cid == 0);
+ /* We are register 2 element, one have root_models and the other have light_model */
+ TEST_ASSERT(dev_comp->elem_count == 2);
+ /* Check the element pointer */
+ TEST_ASSERT(dev_comp->elem);
+ for (int i = 0; i < dev_comp->elem_count; i++) {
+ struct bt_mesh_elem *elem = &dev_comp->elem[i];
+ /* We are register 1 sig model in it and register 1 vendor model in it */
+ TEST_ASSERT((elem->model_count == 1) || (elem->vnd_model_count == 1));
+
+ for (int j = 0; j < elem->model_count; j++) {
+ struct bt_mesh_model *models = &elem->models[j];
+ TEST_ASSERT(models);
+ /* The model id should equal the onoff server model id */
+ TEST_ASSERT(models->id == ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV);
+ /* The elem pointer in the models shouldn't NULL */
+ TEST_ASSERT(models->elem);
+ /* The models op pointer shouldn't NULL */
+ TEST_ASSERT(models->op);
+ TEST_ASSERT(models->op[0].opcode == ESP_BLE_MESH_MODEL_OP_2(0x82, 0x01));
+ TEST_ASSERT(models->op[1].opcode == ESP_BLE_MESH_MODEL_OP_2(0x82, 0x02));
+ TEST_ASSERT(models->op[2].opcode == ESP_BLE_MESH_MODEL_OP_2(0x82, 0x03));
+ TEST_ASSERT(models->op[0].min_len == 0);
+ TEST_ASSERT(models->op[1].min_len == 2);
+ TEST_ASSERT(models->op[2].min_len == 2);
+ }
+
+ for (int k = 0; k < elem->vnd_model_count; k++) {
+ struct bt_mesh_model *vnd_models = &elem->vnd_models[k];
+ TEST_ASSERT(vnd_models);
+ /* The vendor model company id should equal 0x01 */
+ TEST_ASSERT(vnd_models->vnd.company == 0x01);
+ /* The vendor model company id should equal 0 */
+ TEST_ASSERT(vnd_models->vnd.id == 0x7802);
+ /* The elem pointer in the models shouldn't NULL */
+ TEST_ASSERT(vnd_models->elem);
+ /* The models op pointer shouldn't NULL */
+ TEST_ASSERT(vnd_models->op);
+ TEST_ASSERT(vnd_models->op[0].opcode == ESP_BLE_MESH_MODEL_OP_3(0x01, 0x0D01));
+ TEST_ASSERT(vnd_models->op[1].opcode == ESP_BLE_MESH_MODEL_OP_3(0x10, 0x0D10));
+ TEST_ASSERT(vnd_models->op[2].opcode == ESP_BLE_MESH_MODEL_OP_3(0x08, 0x0D08));
+ TEST_ASSERT(vnd_models->op[0].min_len == 1);
+ TEST_ASSERT(vnd_models->op[1].min_len == 1);
+ TEST_ASSERT(vnd_models->op[2].min_len == 1);
+ }
+ }
+}
diff --git a/components/nvs_flash/test_nvs_host/sdkconfig.h b/components/nvs_flash/test_nvs_host/sdkconfig.h
index e69de29bb2..8b13789179 100644
--- a/components/nvs_flash/test_nvs_host/sdkconfig.h
+++ b/components/nvs_flash/test_nvs_host/sdkconfig.h
@@ -0,0 +1 @@
+
diff --git a/docs/en/COPYRIGHT.rst b/docs/en/COPYRIGHT.rst
index e9aa5a4ed8..cb4272af64 100644
--- a/docs/en/COPYRIGHT.rst
+++ b/docs/en/COPYRIGHT.rst
@@ -56,6 +56,7 @@ These third party libraries can be included into the application (firmware) prod
* :component:`Asio `, Copyright (c) 2003-2018 Christopher M. Kohlhoff is licensed under the Boost Software License.
* :component:`ESP-MQTT ` MQTT Package (contiki-mqtt) - Copyright (c) 2014, Stephen Robinson, MQTT-ESP - Tuan PM is licensed under Apache License 2.0.
+* :component:`BLE Mesh ` is adapted from Zephyr Project, Copyright (c) 2017-2018 Intel Corporation and licensed under Apache License 2.0
Build Tools
-----------
@@ -158,3 +159,4 @@ Copyright (C) 2011, ChaN, all right reserved.
.. _spiffs: https://github.com/pellepl/spiffs
.. _asio: https://github.com/chriskohlhoff/asio
.. _mqtt: https://github.com/espressif/esp-mqtt
+.. _zephyr: https://github.com/zephyrproject-rtos/zephyr
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/Makefile b/examples/bluetooth/ble_mesh/ble_mesh_client_model/Makefile
new file mode 100644
index 0000000000..384ad8ed5e
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/Makefile
@@ -0,0 +1,10 @@
+#
+# This is a project Makefile. It is assumed the directory this Makefile resides in is a
+# project subdirectory.
+#
+
+PROJECT_NAME := ble_mesh_client_model
+
+COMPONENT_ADD_INCLUDEDIRS := components/include
+
+include $(IDF_PATH)/make/project.mk
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/README.md b/examples/bluetooth/ble_mesh/ble_mesh_client_model/README.md
new file mode 100644
index 0000000000..38aa2caee1
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/README.md
@@ -0,0 +1,14 @@
+ESP BLE Mesh Client Model Demo
+========================
+
+This demo shows how to use the Generic OnOff Client Model to get/set the generic on/off state. The basic procedures are as follows:
+
+1. Download and run this demo.
+2. Use any app for BLE Mesh to provision this device as well as the device running the Generic OnOff Server demo.
+3. After both onoff client and server devices are provisioned, use UART1 to input the unicast address of the element within the server device.
+4. The Generic OnOff Client will start to get and set Generic OnOff states periodically.
+
+>**Notes:**
+>
+>1. The NetKey index and AppKey index are fixed to 0x0000 in this demo.
+>2. If the client device is re-provisioned, but the server device is not, the first few get/set messages from the client will be treated as replay attacks. To avoid this, both devices should be re-provisioned prior to transmitting messages.
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/Kconfig.projbuild b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/Kconfig.projbuild
new file mode 100644
index 0000000000..acca212403
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/Kconfig.projbuild
@@ -0,0 +1,24 @@
+
+menu "Example Configuration"
+
+choice ESP32_BOARD
+ prompt "Board selection for BLE Mesh"
+ default ESP_WROOM_32_BOARD
+ help
+ Select this option to choose the board for BLE Mesh. The default is ESP32-WROOM-32
+
+config ESP_WROOM_32_BOARD
+ bool "ESP32-WROOM-32"
+
+config ESP_WROVER_BOARD
+ bool "ESP32-WROVER"
+
+endchoice
+
+config HCI_5_0_VERSION
+ bool "This option facilitates sending with 20ms non-connectable interval even if the controller is not HCI 5.0"
+ default y
+ help
+ It is a temporary solution and needs further modifications
+
+endmenu
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/ble_mesh_demo_main.c b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/ble_mesh_demo_main.c
new file mode 100644
index 0000000000..10155067f7
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/ble_mesh_demo_main.c
@@ -0,0 +1,541 @@
+/* main.c - Application main entry point */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+#include
+
+#include "esp_log.h"
+#include "nvs_flash.h"
+
+#include "esp_bt.h"
+#include "esp_bt_main.h"
+#include "esp_bt_device.h"
+#include "esp_ble_mesh_common_api.h"
+#include "esp_ble_mesh_provisioning_api.h"
+#include "esp_ble_mesh_networking_api.h"
+#include "esp_ble_mesh_config_model_api.h"
+#include "esp_ble_mesh_generic_model_api.h"
+
+#include "board.h"
+
+#define TAG "ble_mesh_client"
+
+#define CID_ESP 0x02E5
+
+#define MSG_SEND_TTL 3
+#define MSG_SEND_REL false
+#define MSG_TIMEOUT 0 /* 0 indicates that timeout value from menuconfig will be used */
+#define MSG_ROLE ROLE_NODE
+
+extern struct _led_state led_state[3];
+
+static uint8_t dev_uuid[16] = { 0xdd, 0xdd };
+static uint16_t node_net_idx = ESP_BLE_MESH_KEY_UNUSED;
+static uint16_t node_app_idx = ESP_BLE_MESH_KEY_UNUSED;
+static uint8_t remote_onoff = LED_OFF;
+
+/* The remote node address shall be input through UART1, see board.c */
+uint16_t remote_addr = ESP_BLE_MESH_ADDR_UNASSIGNED;
+
+static esp_ble_mesh_client_t onoff_client;
+
+static esp_ble_mesh_cfg_srv_t config_server = {
+ .relay = ESP_BLE_MESH_RELAY_DISABLED,
+ .beacon = ESP_BLE_MESH_BEACON_ENABLED,
+#if defined(CONFIG_BT_MESH_FRIEND)
+ .friend_state = ESP_BLE_MESH_FRIEND_ENABLED,
+#else
+ .friend_state = ESP_BLE_MESH_FRIEND_NOT_SUPPORTED,
+#endif
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ .gatt_proxy = ESP_BLE_MESH_GATT_PROXY_ENABLED,
+#else
+ .gatt_proxy = ESP_BLE_MESH_GATT_PROXY_NOT_SUPPORTED,
+#endif
+ .default_ttl = 7,
+
+ /* 3 transmissions with 20ms interval */
+ .net_transmit = ESP_BLE_MESH_TRANSMIT(2, 20),
+ .relay_retransmit = ESP_BLE_MESH_TRANSMIT(2, 20),
+};
+
+static esp_ble_mesh_model_pub_t onoff_srv_pub = {
+ .msg = NET_BUF_SIMPLE(2 + 1),
+ .update = NULL,
+ .dev_role = MSG_ROLE,
+};
+
+static esp_ble_mesh_model_pub_t onoff_cli_pub = {
+ .msg = NET_BUF_SIMPLE(2 + 1),
+ .update = NULL,
+ .dev_role = MSG_ROLE,
+};
+
+static esp_ble_mesh_model_op_t onoff_op[] = {
+ { ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_GET, 0, 0},
+ { ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET, 2, 0},
+ { ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK, 2, 0},
+ /* Each model operation struct array must use this terminator
+ * as the end tag of the operation uint. */
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+static esp_ble_mesh_model_t root_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&config_server),
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV, onoff_op,
+ &onoff_srv_pub, &led_state[0]),
+ ESP_BLE_MESH_MODEL_GEN_ONOFF_CLI(&onoff_cli_pub, &onoff_client),
+};
+
+static esp_ble_mesh_elem_t elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, root_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+static esp_ble_mesh_comp_t composition = {
+ .cid = CID_ESP,
+ .elements = elements,
+ .element_count = ARRAY_SIZE(elements),
+};
+
+/* Disable OOB security for SILabs Android app */
+static esp_ble_mesh_prov_t provision = {
+ .uuid = dev_uuid,
+#if 0
+ .output_size = 4,
+ .output_actions = ESP_BLE_MESH_DISPLAY_NUMBER,
+ .input_actions = ESP_BLE_MESH_PUSH,
+ .input_size = 4,
+#else
+ .output_size = 0,
+ .output_actions = 0,
+#endif
+};
+
+static int output_number(esp_ble_mesh_output_action_t action, uint32_t number)
+{
+ board_output_number(action, number);
+ return 0;
+}
+
+static void prov_complete(uint16_t net_idx, uint16_t addr, uint8_t flags, uint32_t iv_index)
+{
+ ESP_LOGI(TAG, "net_idx: 0x%04x, addr: 0x%04x", net_idx, addr);
+ ESP_LOGI(TAG, "flags: 0x%02x, iv_index: 0x%08x", flags, iv_index);
+ board_prov_complete();
+ node_net_idx = net_idx;
+}
+
+static void gen_onoff_get_handler(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx,
+ uint16_t length, uint8_t *data)
+{
+ struct _led_state *led = (struct _led_state *)model->user_data;
+ uint8_t send_data;
+ esp_err_t err;
+
+ ESP_LOGI(TAG, "%s, addr 0x%04x onoff 0x%02x", __func__, model->element->element_addr, led->current);
+
+ send_data = led->current;
+ /* Send Generic OnOff Status as a response to Generic OnOff Get */
+ err = esp_ble_mesh_server_model_send_msg(model, ctx, ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS,
+ sizeof(send_data), &send_data);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "%s: Generic OnOff Status failed", __func__);
+ return;
+ }
+}
+
+static void gen_onoff_set_unack_handler(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx,
+ uint16_t length, uint8_t *data)
+{
+ struct _led_state *led = (struct _led_state *)model->user_data;
+ uint8_t prev_onoff;
+ esp_err_t err;
+
+ ESP_LOGI(TAG, "%s, addr 0x%02x onoff 0x%02x", __func__, model->element->element_addr, led->current);
+
+ prev_onoff = led->previous;
+ led->current = data[0];
+ remote_onoff = led->current;
+
+ board_led_operation(led->pin, led->current);
+
+ /* If Generic OnOff state is changed, and the publish address of Generic OnOff Server
+ * model is valid, Generic OnOff Status will be published.
+ */
+ if (prev_onoff != led->current && model->pub->publish_addr != ESP_BLE_MESH_ADDR_UNASSIGNED) {
+ ESP_LOGI(TAG, "Publish previous 0x%02x current 0x%02x", prev_onoff, led->current);
+ err = esp_ble_mesh_model_publish(model, ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS,
+ sizeof(led->current), &led->current, ROLE_NODE);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "%s: Publish Generic OnOff Status failed", __func__);
+ return;
+ }
+ }
+}
+
+static void gen_onoff_set_handler(esp_ble_mesh_model_t *model,
+ esp_ble_mesh_msg_ctx_t *ctx,
+ uint16_t length, uint8_t *data)
+{
+ struct net_buf_simple *msg = model->pub->msg;
+ struct _led_state *led = (struct _led_state *)model->user_data;
+ uint8_t prev_onoff, send_data;
+ esp_err_t err;
+
+ ESP_LOGI(TAG, "%s, addr 0x%02x onoff 0x%02x", __func__, model->element->element_addr, led->current);
+
+ prev_onoff = led->previous;
+ led->current = data[0];
+ remote_onoff = led->current;
+
+ board_led_operation(led->pin, led->current);
+
+ send_data = led->current;
+ /* Send Generic OnOff Status as a response to Generic OnOff Get */
+ err = esp_ble_mesh_server_model_send_msg(model, ctx, ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS,
+ sizeof(send_data), &send_data);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "%s: Generic OnOff Status failed", __func__);
+ return;
+ }
+
+ /* If Generic OnOff state is changed, and the publish address of Generic OnOff Server
+ * model is valid, Generic OnOff Status will be published.
+ */
+ if (prev_onoff != led->current && model->pub->publish_addr != ESP_BLE_MESH_ADDR_UNASSIGNED) {
+ ESP_LOGI(TAG, "Publish previous 0x%02x current 0x%02x", prev_onoff, led->current);
+ bt_mesh_model_msg_init(msg, ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS);
+ err = esp_ble_mesh_model_publish(model, ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS,
+ sizeof(send_data), &send_data, ROLE_NODE);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "%s: Publish Generic OnOff Status failed", __func__);
+ return;
+ }
+ }
+}
+
+static char *esp_ble_mesh_prov_event_to_str(esp_ble_mesh_prov_cb_event_t event)
+{
+ switch (event) {
+ case ESP_BLE_MESH_PROV_REGISTER_COMP_EVT:
+ return "ESP_BLE_MESH_PROV_REGISTER_COMP_EVT";
+ case ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT";
+ case ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT";
+ case ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT";
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_NUMBER_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_OUTPUT_NUMBER_EVT";
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT";
+ case ESP_BLE_MESH_NODE_PROV_INPUT_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_INPUT_EVT";
+ case ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT";
+ case ESP_BLE_MESH_NODE_PROV_RESET_EVT:
+ return "ESP_BLE_MESH_NODE_PROV_RESET_EVT";
+ default:
+ return "Invalid BLE Mesh provision event";
+ }
+
+ return NULL;
+}
+
+static void esp_ble_mesh_prov_cb(esp_ble_mesh_prov_cb_event_t event,
+ esp_ble_mesh_prov_cb_param_t *param)
+{
+ ESP_LOGI(TAG, "%s, event = %s", __func__, esp_ble_mesh_prov_event_to_str(event));
+
+ switch (event) {
+ case ESP_BLE_MESH_PROV_REGISTER_COMP_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_PROV_REGISTER_COMP_EVT, err_code %d", param->prov_register_comp.err_code);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT, err_code %d", param->node_prov_enable_comp.err_code);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT, bearer %s",
+ param->node_prov_link_open.bearer == ESP_BLE_MESH_PROV_ADV ? "PB-ADV" : "PB-GATT");
+ break;
+ case ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT, bearer %s",
+ param->node_prov_link_close.bearer == ESP_BLE_MESH_PROV_ADV ? "PB-ADV" : "PB-GATT");
+ break;
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_NUMBER_EVT:
+ output_number(param->node_prov_output_num.action, param->node_prov_output_num.number);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT:
+ break;
+ case ESP_BLE_MESH_NODE_PROV_INPUT_EVT:
+ break;
+ case ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT:
+ prov_complete(param->node_prov_complete.net_idx, param->node_prov_complete.addr,
+ param->node_prov_complete.flags, param->node_prov_complete.iv_index);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_RESET_EVT:
+ break;
+ case ESP_BLE_MESH_NODE_SET_UNPROV_DEV_NAME_COMP_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_NODE_SET_UNPROV_DEV_NAME_COMP_EVT, err_code %d", param->node_set_unprov_dev_name_comp.err_code);
+ break;
+ default:
+ break;
+ }
+
+ return;
+}
+
+static esp_err_t esp_ble_mesh_set_msg_common(esp_ble_mesh_client_common_param_t *common,
+ esp_ble_mesh_generic_client_set_state_t *set,
+ esp_ble_mesh_model_t *model, uint32_t opcode,
+ uint8_t onoff)
+{
+ if (!common || !set || !model) {
+ return ESP_ERR_INVALID_ARG;
+ }
+
+ common->opcode = opcode;
+ common->model = model;
+ common->ctx.net_idx = node_net_idx;
+ common->ctx.app_idx = node_app_idx;
+ common->ctx.addr = remote_addr;
+ common->ctx.send_ttl = MSG_SEND_TTL;
+ common->ctx.send_rel = MSG_SEND_REL;
+ common->msg_timeout = MSG_TIMEOUT;
+ common->msg_role = MSG_ROLE;
+ set->onoff_set.op_en = false;
+ set->onoff_set.onoff = onoff;
+ set->onoff_set.tid = 0x0;
+
+ return ESP_OK;
+}
+
+
+static void esp_ble_mesh_model_cb(esp_ble_mesh_model_cb_event_t event,
+ esp_ble_mesh_model_cb_param_t *param)
+{
+ esp_ble_mesh_client_common_param_t common = {0};
+ int err;
+
+ switch (event) {
+ case ESP_BLE_MESH_MODEL_OPERATION_EVT: {
+ if (!param->model_operation.model || !param->model_operation.model->op || !param->model_operation.ctx) {
+ ESP_LOGE(TAG, "ESP_BLE_MESH_MODEL_OPERATION_EVT parameter is NULL");
+ return;
+ }
+ if (node_app_idx == ESP_BLE_MESH_KEY_UNUSED) {
+ /* Generic OnOff Server/Client Models need to bind with the same app key */
+ node_app_idx = param->model_operation.ctx->app_idx;
+ }
+ switch (param->model_operation.opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_GET:
+ gen_onoff_get_handler(param->model_operation.model, param->model_operation.ctx,
+ param->model_operation.length, param->model_operation.msg);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET: {
+ esp_ble_mesh_generic_client_set_state_t set_state = {0};
+ gen_onoff_set_handler(param->model_operation.model, param->model_operation.ctx,
+ param->model_operation.length, param->model_operation.msg);
+ /* This node has a Generic OnOff Client and Server Model.
+ * When Generic OnOff Server Model receives a Generic OnOff Set message, after
+ * this message is been handled, the Generic OnOff Client Model will send the
+ * Generic OnOff Set message to another node(contains Generic OnOff Server Model)
+ * identified by the remote_addr.
+ */
+ esp_ble_mesh_set_msg_common(&common, &set_state, onoff_client.model,
+ ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET, remote_onoff);
+ err = esp_ble_mesh_generic_client_set_state(&common, &set_state);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "%s: Generic OnOff Set failed", __func__);
+ return;
+ }
+ break;
+ }
+ case ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK: {
+ esp_ble_mesh_generic_client_set_state_t set_state = {0};
+ gen_onoff_set_unack_handler(param->model_operation.model, param->model_operation.ctx,
+ param->model_operation.length, param->model_operation.msg);
+ /* This node has a Generic OnOff Client and Server Model.
+ * When Generic OnOff Client Model receives a Generic OnOff Set Unack message,
+ * after this message is been handled, the Generic OnOff Client Model will send
+ * the Generic OnOff Set Unack message to another node(contains Generic OnOff
+ * Server Model) identified by the remote_addr.
+ */
+ esp_ble_mesh_set_msg_common(&common, &set_state, onoff_client.model,
+ ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK, remote_onoff);
+ err = esp_ble_mesh_generic_client_set_state(&common, &set_state);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "%s: Generic OnOff Set Unack failed", __func__);
+ return;
+ }
+ break;
+ }
+ default:
+ break;
+ }
+ break;
+ }
+ case ESP_BLE_MESH_MODEL_SEND_COMP_EVT:
+ break;
+ case ESP_BLE_MESH_MODEL_PUBLISH_COMP_EVT:
+ break;
+ default:
+ break;
+ }
+ return;
+}
+
+static void esp_ble_mesh_generic_cb(esp_ble_mesh_generic_client_cb_event_t event,
+ esp_ble_mesh_generic_client_cb_param_t *param)
+{
+ uint32_t opcode;
+ int err;
+
+ ESP_LOGI(TAG, "%s: event is %d, error code is %d, opcode is 0x%x",
+ __func__, event, param->error_code, param->params->opcode);
+
+ opcode = param->params->opcode;
+
+ switch (event) {
+ case ESP_BLE_MESH_GENERIC_CLIENT_GET_STATE_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_GENERIC_CLIENT_GET_STATE_EVT");
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_GET:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_GET: onoff %d", param->status_cb.onoff_status.present_onoff);
+ break;
+ default:
+ break;
+ }
+ break;
+ case ESP_BLE_MESH_GENERIC_CLIENT_SET_STATE_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_GENERIC_CLIENT_SET_STATE_EVT");
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET: onoff %d", param->status_cb.onoff_status.present_onoff);
+ break;
+ default:
+ break;
+ }
+ break;
+ case ESP_BLE_MESH_GENERIC_CLIENT_PUBLISH_EVT:
+ ESP_LOGI(TAG, "ESP_BLE_MESH_GENERIC_CLIENT_PUBLISH_EVT");
+ break;
+ case ESP_BLE_MESH_GENERIC_CLIENT_TIMEOUT_EVT: {
+ ESP_LOGI(TAG, "ESP_BLE_MESH_GENERIC_CLIENT_TIMEOUT_EVT");
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET: {
+ esp_ble_mesh_client_common_param_t common = {0};
+ esp_ble_mesh_generic_client_set_state_t set_state = {0};
+ /* If failed to get the response of Generic OnOff Set, resend Generic OnOff Set */
+ esp_ble_mesh_set_msg_common(&common, &set_state, onoff_client.model,
+ ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET, remote_onoff);
+ err = esp_ble_mesh_generic_client_set_state(&common, &set_state);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "%s: Generic OnOff Set failed", __func__);
+ return;
+ }
+ break;
+ }
+ default:
+ break;
+ }
+ break;
+ }
+ default:
+ break;
+ }
+ return;
+}
+
+static int ble_mesh_init(void)
+{
+ int err = 0;
+
+ memcpy(dev_uuid + 2, esp_bt_dev_get_address(), ESP_BD_ADDR_LEN);
+
+ esp_ble_mesh_register_prov_callback(esp_ble_mesh_prov_cb);
+ esp_ble_mesh_register_custom_model_callback(esp_ble_mesh_model_cb);
+ esp_ble_mesh_register_generic_client_callback(esp_ble_mesh_generic_cb);
+
+ err = esp_ble_mesh_init(&provision, &composition);
+ if (err) {
+ ESP_LOGE(TAG, "Initializing mesh failed (err %d)", err);
+ return err;
+ }
+
+ esp_ble_mesh_node_prov_enable(ESP_BLE_MESH_PROV_ADV | ESP_BLE_MESH_PROV_GATT);
+
+ ESP_LOGI(TAG, "BLE Mesh Node initialized");
+
+ board_led_operation(LED_G, LED_ON);
+
+ return err;
+}
+
+static esp_err_t bluetooth_init(void)
+{
+ esp_err_t ret;
+
+ ret = nvs_flash_init();
+ if (ret == ESP_ERR_NVS_NO_FREE_PAGES) {
+ ESP_ERROR_CHECK(nvs_flash_erase());
+ ret = nvs_flash_init();
+ }
+ ESP_ERROR_CHECK(ret);
+
+ ESP_ERROR_CHECK(esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT));
+
+ esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
+ ret = esp_bt_controller_init(&bt_cfg);
+ if (ret) {
+ ESP_LOGE(TAG, "%s initialize controller failed", __func__);
+ return ret;
+ }
+
+ ret = esp_bt_controller_enable(ESP_BT_MODE_BLE);
+ if (ret) {
+ ESP_LOGE(TAG, "%s enable controller failed", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_init();
+ if (ret) {
+ ESP_LOGE(TAG, "%s init bluetooth failed", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_enable();
+ if (ret) {
+ ESP_LOGE(TAG, "%s enable bluetooth failed", __func__);
+ return ret;
+ }
+
+ return ret;
+}
+
+void app_main(void)
+{
+ int err;
+
+ ESP_LOGI(TAG, "Initializing...");
+
+ board_init();
+
+ err = bluetooth_init();
+ if (err) {
+ ESP_LOGE(TAG, "esp32_bluetooth_init failed (err %d)", err);
+ return;
+ }
+
+ /* Initialize the Bluetooth Mesh Subsystem */
+ err = ble_mesh_init();
+ if (err) {
+ ESP_LOGE(TAG, "Bluetooth mesh init failed (err %d)", err);
+ }
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/board.c b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/board.c
new file mode 100644
index 0000000000..d81bc58815
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/board.c
@@ -0,0 +1,115 @@
+/* board.c - Board-specific hooks */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ * Additional Copyright (c) 2018 Espressif Systems (Shanghai) PTE LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "freertos/event_groups.h"
+
+#include "driver/uart.h"
+#include "esp_log.h"
+
+#include "esp_ble_mesh_provisioning_api.h"
+#include "board.h"
+
+#define TAG "BOARD"
+
+#define MESH_UART_NUM UART_NUM_1
+#define MESH_UART (&UART1)
+
+#define UART_BUF_SIZE 128
+
+#define UART1_TX_PIN GPIO_NUM_16
+#define UART1_RX_PIN GPIO_NUM_17
+
+extern uint16_t remote_addr;
+
+struct _led_state led_state[3] = {
+ { LED_OFF, LED_OFF, LED_R, "red" },
+ { LED_OFF, LED_OFF, LED_G, "green" },
+ { LED_OFF, LED_OFF, LED_B, "blue" },
+};
+
+void board_output_number(esp_ble_mesh_output_action_t action, uint32_t number)
+{
+ ESP_LOGI(TAG, "Board output number %d", number);
+}
+
+void board_prov_complete(void)
+{
+ board_led_operation(LED_G, LED_OFF);
+}
+
+void board_led_operation(uint8_t pin, uint8_t onoff)
+{
+ for (int i = 0; i < ARRAY_SIZE(led_state); i++) {
+ if (led_state[i].pin != pin) {
+ continue;
+ }
+ if (onoff == led_state[i].previous) {
+ ESP_LOGW(TAG, "led %s is already %s",
+ led_state[i].name, (onoff ? "on" : "off"));
+ return;
+ }
+ gpio_set_level(pin, onoff);
+ led_state[i].previous = onoff;
+ return;
+ }
+ ESP_LOGE(TAG, "LED is not found!");
+}
+
+static void board_uart_task(void *p)
+{
+ uint8_t *data = calloc(1, UART_BUF_SIZE);
+ uint32_t input;
+
+ while (1) {
+ int len = uart_read_bytes(MESH_UART_NUM, data, UART_BUF_SIZE, 100 / portTICK_RATE_MS);
+ if (len > 0) {
+ input = strtoul((const char *)data, NULL, 16);
+ remote_addr = input & 0xFFFF;
+ ESP_LOGI(TAG, "%s: input 0x%08x, remote_addr 0x%04x", __func__, input, remote_addr);
+ memset(data, 0, UART_BUF_SIZE);
+ }
+ }
+
+ vTaskDelete(NULL);
+}
+
+static void board_led_init(void)
+{
+ for (int i = 0; i < ARRAY_SIZE(led_state); i++) {
+ gpio_pad_select_gpio(led_state[i].pin);
+ gpio_set_direction(led_state[i].pin, GPIO_MODE_OUTPUT);
+ gpio_set_level(led_state[i].pin, LED_OFF);
+ led_state[i].previous = LED_OFF;
+ }
+}
+
+static void board_uart_init(void)
+{
+ uart_config_t uart_config = {
+ .baud_rate = 115200,
+ .data_bits = UART_DATA_8_BITS,
+ .parity = UART_PARITY_DISABLE,
+ .stop_bits = UART_STOP_BITS_1,
+ .flow_ctrl = UART_HW_FLOWCTRL_DISABLE
+ };
+ uart_param_config(MESH_UART_NUM, &uart_config);
+ uart_set_pin(MESH_UART_NUM, UART1_TX_PIN, UART1_RX_PIN, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
+ uart_driver_install(MESH_UART_NUM, UART_BUF_SIZE * 2, 0, 0, NULL, 0);
+}
+
+void board_init(void)
+{
+ board_led_init();
+ board_uart_init();
+ xTaskCreate(board_uart_task, "board_uart_task", 2048, NULL, 5, NULL);
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/board.h b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/board.h
new file mode 100644
index 0000000000..c30626a997
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/board.h
@@ -0,0 +1,41 @@
+/* board.h - Board-specific hooks */
+
+/*
+ * Copyright (c) 2017 Intel Corporation
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#ifndef _BOARD_H_
+#define _BOARD_H_
+
+#include "esp_ble_mesh_defs.h"
+
+#if defined(CONFIG_ESP_WROOM_32_BOARD)
+#define LED_R GPIO_NUM_25
+#define LED_G GPIO_NUM_26
+#define LED_B GPIO_NUM_27
+#elif defined(CONFIG_ESP_WROVER_BOARD)
+#define LED_R GPIO_NUM_0
+#define LED_G GPIO_NUM_2
+#define LED_B GPIO_NUM_4
+#endif
+
+#define LED_ON 1
+#define LED_OFF 0
+
+struct _led_state {
+ uint8_t current;
+ uint8_t previous;
+ uint8_t pin;
+ char *name;
+};
+
+void board_output_number(esp_ble_mesh_output_action_t action, uint32_t number);
+
+void board_prov_complete(void);
+
+void board_led_operation(uint8_t pin, uint8_t onoff);
+
+void board_init(void);
+
+#endif
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/component.mk b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/component.mk
new file mode 100644
index 0000000000..a98f634eae
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/main/component.mk
@@ -0,0 +1,4 @@
+#
+# "main" pseudo-component makefile.
+#
+# (Uses default behaviour of compiling all source files in directory, adding 'include' to include path.)
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_client_model/sdkconfig.defaults b/examples/bluetooth/ble_mesh/ble_mesh_client_model/sdkconfig.defaults
new file mode 100644
index 0000000000..ccec2ba49e
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_client_model/sdkconfig.defaults
@@ -0,0 +1,32 @@
+# Override some defaults so BT stack is enabled
+# by default in this example
+CONFIG_BT_ENABLED=y
+CONFIG_BT_MESH=y
+CONFIG_BT_MESH_NODE=y
+CONFIG_BT_MESH_PROV=y
+CONFIG_NET_BUF_POOL_USAGE=y
+CONFIG_BT_MESH_PROXY=y
+CONFIG_BT_MESH_PB_GATT=y
+CONFIG_BT_MESH_GATT_PROXY=y
+CONFIG_BT_MESH_NODE_ID_TIMEOUT=60
+CONFIG_BT_MESH_PROXY_FILTER_SIZE=1
+CONFIG_BT_MESH_SELF_TEST=
+CONFIG_BT_MESH_IV_UPDATE_TEST=
+CONFIG_BT_MESH_SUBNET_COUNT=1
+CONFIG_BT_MESH_APP_KEY_COUNT=1
+CONFIG_BT_MESH_MODEL_KEY_COUNT=1
+CONFIG_BT_MESH_MODEL_GROUP_COUNT=1
+CONFIG_BT_MESH_LABEL_COUNT=1
+CONFIG_BT_MESH_CRPL=10
+CONFIG_BT_MESH_MSG_CACHE_SIZE=10
+CONFIG_BT_MESH_ADV_BUF_COUNT=20
+CONFIG_BT_MESH_TX_SEG_MSG_COUNT=6
+CONFIG_BT_MESH_RX_SEG_MSG_COUNT=1
+CONFIG_BT_MESH_RX_SDU_MAX=384
+CONFIG_BT_MESH_RELAY=y
+CONFIG_BT_MESH_LOW_POWER=
+CONFIG_BT_MESH_FRIEND=
+CONFIG_BT_MESH_CFG_CLI=
+CONFIG_BT_MESH_HEALTH_CLI=
+CONFIG_BT_MESH_GENERIC_ONOFF_CLI=y
+CONFIG_BTU_TASK_STACK_SIZE=4512
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/Makefile b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/Makefile
new file mode 100644
index 0000000000..209aa2e0cc
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/Makefile
@@ -0,0 +1,10 @@
+#
+# This is a project Makefile. It is assumed the directory this Makefile resides in is a
+# project subdirectory.
+#
+
+PROJECT_NAME := ble_mesh_console_node
+
+COMPONENT_ADD_INCLUDEDIRS := components/include
+
+include $(IDF_PATH)/make/project.mk
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/README.md b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/README.md
new file mode 100644
index 0000000000..0bb260c816
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/README.md
@@ -0,0 +1,9 @@
+# ble mesh node console demo
+## Introduction
+This demo implements ble mesh node basic features.Based on this demo, node can be scaned and proved by provisioner, reply get/set message to provisioner.
+
+Demo steps:
+1. Build the ble mesh node console demo with sdkconfig.default
+2. register node and set oob info, load model to init ble mesh node
+3. enable bearer, so that it can be scaned and provisioned by provisioner
+
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_adapter.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_adapter.c
new file mode 100644
index 0000000000..47622b9d1d
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_adapter.c
@@ -0,0 +1,164 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "ble_mesh_adapter.h"
+
+esp_ble_mesh_model_t *ble_mesh_get_model(uint16_t model_id)
+{
+ esp_ble_mesh_model_t *model = NULL;
+
+ switch (model_id) {
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_SRV:
+ model = &config_server_models[0];
+ break;
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_CLI:
+ model = &config_client_models[0];
+ break;
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV:
+ model = &gen_onoff_srv_models[1];
+ break;
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_CLI:
+ model = &gen_onoff_cli_models[1];
+ break;
+#endif
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI:
+ model = &test_perf_cli_models[0];
+ break;
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV:
+ model = &test_perf_srv_models[0];
+ break;
+ }
+ return model;
+}
+
+esp_ble_mesh_comp_t *ble_mesh_get_component(uint16_t model_id)
+{
+ esp_ble_mesh_comp_t *comp = NULL;
+
+ switch (model_id) {
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_SRV:
+ comp = &config_server_comp;
+ break;
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_CLI:
+ comp = &config_client_comp;
+ break;
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV:
+ comp = &gen_onoff_srv_comp;
+ break;
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_CLI:
+ comp = &gen_onoff_cli_comp;
+ break;
+#endif
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI:
+ comp = &test_perf_cli_comp;
+ break;
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV:
+ comp = &test_perf_srv_comp;
+ break;
+ }
+ return comp;
+}
+
+void ble_mesh_node_init()
+{
+ uint16_t i;
+
+ for (i = 0; i < NODE_MAX_GROUP_CONFIG; i++) {
+ ble_mesh_node_prestore_params[i].net_idx = 0xFFFF;
+ ble_mesh_node_prestore_params[i].unicast_addr = 0xFFFF;
+ }
+
+ ble_mesh_node_sema = xSemaphoreCreateMutex();
+ if (!ble_mesh_node_sema) {
+ ESP_LOGE(TAG, "%s init fail, mesh node semaphore create fail", __func__);
+ }
+}
+
+void ble_mesh_set_node_prestore_params(uint16_t netkey_index, uint16_t unicast_addr)
+{
+ uint16_t i;
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY);
+ for (i = 0; i < NODE_MAX_GROUP_CONFIG; i++) {
+ if (ble_mesh_node_prestore_params[i].net_idx != 0xFFFF && ble_mesh_node_prestore_params[i].unicast_addr != 0xFFFF) {
+ ble_mesh_node_prestore_params[i].net_idx = netkey_index;
+ ble_mesh_node_prestore_params[i].unicast_addr = unicast_addr;
+ }
+ }
+ xSemaphoreGive(ble_mesh_node_sema);
+}
+
+void ble_mesh_node_statistics_get()
+{
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY);
+ ESP_LOGI(TAG, "statistics:%d,%d\n", ble_mesh_node_statistics.statistics, ble_mesh_node_statistics.package_num);
+ xSemaphoreGive(ble_mesh_node_sema);
+}
+
+int ble_mesh_node_statistics_accumultate(uint8_t *data, uint32_t value, uint16_t type)
+{
+ uint16_t i;
+ uint16_t sequence_num = (data[0] << 8) | data[1];
+
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY);
+ for (i = 0; i < ble_mesh_node_statistics.total_package_num; i++) {
+ if (ble_mesh_node_statistics.package_index[i] == sequence_num) {
+ xSemaphoreGive(ble_mesh_node_sema);
+ return 1;
+ }
+ }
+
+ // package type wrong
+ if (data[2] != type) {
+ xSemaphoreGive(ble_mesh_node_sema);
+ return 1;
+ }
+
+ for (i = 0; i < ble_mesh_node_statistics.total_package_num; i++) {
+ if (ble_mesh_node_statistics.package_index[i] == 0) {
+ ble_mesh_node_statistics.package_index[i] = sequence_num;
+ ble_mesh_node_statistics.package_num += 1;
+ ble_mesh_node_statistics.statistics += value;
+ break;
+ }
+ }
+ xSemaphoreGive(ble_mesh_node_sema);
+ return 0;
+}
+
+int ble_mesh_node_statistics_init(uint16_t package_num)
+{
+ uint16_t i;
+
+ ble_mesh_node_statistics.package_index = malloc(sizeof(uint16_t) * package_num);
+ ble_mesh_node_statistics.total_package_num = package_num;
+ if (ble_mesh_node_statistics.package_index == NULL) {
+ ESP_LOGE(TAG, " %s, %d malloc fail\n", __func__, __LINE__);
+ return 1;
+ }
+
+ ble_mesh_node_statistics.package_num = 0;
+ for (i = 0; i < package_num; i++) {
+ ble_mesh_node_statistics.package_index[i] = 0;
+ }
+ return 0;
+}
+
+void ble_mesh_node_statistics_destroy()
+{
+ if (ble_mesh_node_statistics.package_index != NULL) {
+ free(ble_mesh_node_statistics.package_index);
+ }
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_adapter.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_adapter.h
new file mode 100644
index 0000000000..a5122336e2
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_adapter.h
@@ -0,0 +1,97 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BLE_MESH_ADAPTER_H_
+#define _BLE_MESH_ADAPTER_H_
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/semphr.h"
+
+#include "ble_mesh_console_lib.h"
+#include "ble_mesh_cfg_srv_model.h"
+
+#define TAG "ble_mesh_node_console"
+
+typedef enum {
+ VENDOR_MODEL_PERF_OPERATION_TYPE_GET = 1,
+ VENDOR_MODEL_PERF_OPERATION_TYPE_SET,
+ VENDOR_MODEL_PERF_OPERATION_TYPE_SET_UNACK
+} ble_mesh_perf_operation_type;
+
+typedef struct {
+ uint8_t current;
+ uint8_t previous;
+ char *name;
+} ble_mesh_node_status;
+
+typedef struct {
+ uint32_t statistics;
+ uint32_t package_num;
+ uint16_t *package_index;
+ uint32_t total_package_num;
+} ble_mesh_node_statistics_t;
+ble_mesh_node_statistics_t ble_mesh_node_statistics;
+
+extern SemaphoreHandle_t ble_mesh_node_sema;
+
+#define arg_int_to_value(src_msg, dst_msg, message) do { \
+ if (src_msg->count != 0) {\
+ ESP_LOGD(TAG, "\n%s, %s\n", __func__, message);\
+ dst_msg = src_msg->ival[0];\
+ } \
+} while(0) \
+
+#define ble_mesh_node_get_value(index, key, value) do { \
+ uint16_t _index = 0; \
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY); \
+ for (_index = 0; _index < NODE_MAX_GROUP_CONFIG; _index) { \
+ if (node_set_prestore_params[_index].key == value) { \
+ break; \
+ } \
+ } \
+ index = _index; \
+ xSemaphoreGive(ble_mesh_node_sema); \
+} while(0) \
+
+#define ble_mesh_node_set_state(status) do { \
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY); \
+ node_status.previous = node_status.current; \
+ node_status.current = status; \
+ xSemaphoreGive(ble_mesh_node_sema); \
+}while(0) \
+
+#define ble_mesh_node_get_state(status) do { \
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY); \
+ status = node_status.current; \
+ xSemaphoreGive(ble_mesh_node_sema); \
+}while(0) \
+
+#define ble_mesh_callback_check_err_code(err_code, message) do { \
+ if (err_code == ESP_OK) { \
+ ESP_LOGI(TAG, "%s,OK\n", message); \
+ } else { \
+ ESP_LOGE(TAG, "%s,Fail,%d\n", message, err_code); \
+ } \
+}while(0) \
+
+void ble_mesh_node_init();
+void ble_mesh_set_node_prestore_params(uint16_t netkey_index, uint16_t unicast_addr);
+esp_ble_mesh_model_t *ble_mesh_get_model(uint16_t model_id);
+esp_ble_mesh_comp_t *ble_mesh_get_component(uint16_t model_id);
+void ble_mesh_node_statistics_get();
+int ble_mesh_node_statistics_accumultate(uint8_t *data, uint32_t value, uint16_t type);
+int ble_mesh_node_statistics_init(uint16_t package_num);
+void ble_mesh_node_statistics_destroy();
+
+#endif //_BLE_MESH_ADAOTER_H_
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_cfg_srv_model.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_cfg_srv_model.c
new file mode 100644
index 0000000000..695ce2faa0
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_cfg_srv_model.c
@@ -0,0 +1,212 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "ble_mesh_cfg_srv_model.h"
+
+uint8_t dev_uuid[16] = {0xdd, 0xdd};
+
+#if CONFIG_BT_MESH_NODE
+esp_ble_mesh_prov_t prov = {
+ .uuid = dev_uuid,
+};
+#endif //CONFIG_BT_MESH_NODE
+
+#if CONFIG_BT_MESH_PROVISIONER
+esp_ble_mesh_prov_t prov = {
+ .prov_uuid = dev_uuid,
+ .prov_unicast_addr = 0x0001,
+ .prov_start_address = 0x0005,
+ .prov_attention = 0x00,
+ .prov_algorithm = 0x00,
+ .prov_pub_key_oob = 0x00,
+ .prov_pub_key_oob_cb = NULL,
+ .prov_static_oob_val = NULL,
+ .prov_static_oob_len = 0x00,
+ .prov_input_num = NULL,
+ .prov_output_num = NULL,
+ .flags = 0x00,
+ .iv_index = 0x00,
+};
+#endif //CONFIG_BT_MESH_PROVISIONER
+
+esp_ble_mesh_model_pub_t model_pub_config = {
+ .msg = NET_BUF_SIMPLE(2 + 1),
+ .update = NULL,
+ .dev_role = ROLE_NODE,
+};
+
+esp_ble_mesh_model_pub_t vendor_model_pub_config;
+
+// configure server module
+esp_ble_mesh_cfg_srv_t cfg_srv = {
+ .relay = ESP_BLE_MESH_RELAY_ENABLED,
+ .beacon = ESP_BLE_MESH_BEACON_ENABLED,
+#if defined(CONFIG_BT_MESH_FRIEND)
+ .friend_state = BT_MESH_FRIEND_ENABLED,
+#else
+ .friend_state = BT_MESH_FRIEND_NOT_SUPPORTED,
+#endif
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ .gatt_proxy = BT_MESH_GATT_PROXY_ENABLED,
+#else
+ .gatt_proxy = BT_MESH_GATT_PROXY_NOT_SUPPORTED,
+#endif
+ .default_ttl = 7,
+
+ /* 3 transmissions with 20ms interval */
+ .net_transmit = ESP_BLE_MESH_TRANSMIT(2, 20),
+ .relay_retransmit = BT_MESH_TRANSMIT(0, 20),
+};
+
+esp_ble_mesh_model_t config_server_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+};
+
+esp_ble_mesh_elem_t config_server_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_server_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t config_server_comp = {
+ .cid = CID_ESP,
+ .elements = config_server_elements,
+ .element_count = ARRAY_SIZE(config_server_elements),
+};
+
+// config client model
+esp_ble_mesh_model_t config_client_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_MODEL_CFG_CLI(&cfg_cli),
+};
+
+esp_ble_mesh_elem_t config_client_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_client_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t config_client_comp = {
+ .cid = CID_ESP,
+ .elements = config_client_elements,
+ .element_count = ARRAY_SIZE(config_client_elements),
+};
+
+// configure special module
+esp_ble_mesh_model_op_t gen_onoff_srv_model_op_config[] = {
+ {ESP_BLE_MESH_MODEL_OP_2(0x82, 0x01), 0, 0},
+ {ESP_BLE_MESH_MODEL_OP_2(0x82, 0x02), 2, 0},
+ {ESP_BLE_MESH_MODEL_OP_2(0x82, 0x03), 2, 0},
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+esp_ble_mesh_model_t gen_onoff_srv_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV, gen_onoff_srv_model_op_config, &model_pub_config, NULL),
+};
+
+esp_ble_mesh_elem_t gen_onoff_srv_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, gen_onoff_srv_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t gen_onoff_srv_comp = {
+ .cid = CID_ESP,
+ .elements = gen_onoff_srv_elements,
+ .element_count = ARRAY_SIZE(gen_onoff_srv_elements),
+};
+
+// config generic onoff client
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+
+esp_ble_mesh_client_t gen_onoff_cli;
+
+esp_ble_mesh_model_t gen_onoff_cli_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_MODEL_CFG_CLI(&cfg_cli),
+ ESP_BLE_MESH_MODEL_GEN_ONOFF_CLI(&model_pub_config, &gen_onoff_cli),
+};
+
+esp_ble_mesh_elem_t gen_onoff_cli_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, gen_onoff_cli_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t gen_onoff_cli_comp = {
+ .cid = CID_ESP,
+ .elements = gen_onoff_cli_elements,
+ .element_count = ARRAY_SIZE(gen_onoff_cli_elements),
+};
+#endif //CONFIG_BT_MESH_GENERIC_ONOFF_CLI
+
+//CONFIG VENDOR MODEL TEST PERFORMANCE
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV 0x2000
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI 0x2001
+
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET ESP_BLE_MESH_MODEL_OP_3(0x01, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET ESP_BLE_MESH_MODEL_OP_3(0x02, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK ESP_BLE_MESH_MODEL_OP_3(0x03, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS ESP_BLE_MESH_MODEL_OP_3(0x04, CID_ESP)
+
+esp_ble_mesh_client_op_pair_t test_perf_cli_op_pair[] = {
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET, ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET, ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK, ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS},
+};
+
+esp_ble_mesh_client_t test_perf_cli = {
+ .op_pair_size = ARRAY_SIZE(test_perf_cli_op_pair),
+ .op_pair = test_perf_cli_op_pair,
+};
+
+esp_ble_mesh_model_op_t test_perf_srv_op[] = {
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET, 1, 0},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET, 1, 0},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK, 1, 0},
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+esp_ble_mesh_model_op_t test_perf_cli_op[] = {
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS, 1, 0},
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+esp_ble_mesh_model_t config_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_MODEL_CFG_CLI(&cfg_cli),
+};
+
+esp_ble_mesh_model_t test_perf_cli_models[] = {
+ ESP_BLE_MESH_VENDOR_MODEL(CID_ESP, ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI,
+ test_perf_cli_op, &vendor_model_pub_config, &test_perf_cli),
+};
+
+esp_ble_mesh_elem_t test_perf_cli_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_models, test_perf_cli_models),
+};
+
+esp_ble_mesh_comp_t test_perf_cli_comp = {
+ .cid = CID_ESP,
+ .elements = test_perf_cli_elements,
+ .element_count = ARRAY_SIZE(test_perf_cli_elements),
+};
+
+esp_ble_mesh_model_t test_perf_srv_models[] = {
+ ESP_BLE_MESH_VENDOR_MODEL(CID_ESP, ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV,
+ test_perf_srv_op, NULL, NULL),
+};
+
+esp_ble_mesh_elem_t test_perf_srv_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_models, test_perf_srv_models),
+};
+
+esp_ble_mesh_comp_t test_perf_srv_comp = {
+ .cid = CID_ESP,
+ .elements = test_perf_srv_elements,
+ .element_count = ARRAY_SIZE(test_perf_srv_elements),
+};
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_cfg_srv_model.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_cfg_srv_model.h
new file mode 100644
index 0000000000..72f012506f
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_cfg_srv_model.h
@@ -0,0 +1,109 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BLE_MESH_CFG_SRV_MODEL_H_
+#define _BLE_MESH_CFG_SRV_MODEL_H_
+
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_config_model_api.h"
+
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+#include "esp_ble_mesh_generic_model_api.h"
+#endif //CONFIG_BT_MESH_GENERIC_ONOFF_CLI
+
+#define NODE_MAX_GROUP_CONFIG 3
+#define CID_ESP 0x02C4
+
+extern uint8_t dev_uuid[16];
+
+typedef struct {
+ uint16_t net_idx;
+ uint16_t unicast_addr;
+} ble_mesh_node_config_params;
+ble_mesh_node_config_params ble_mesh_node_prestore_params[NODE_MAX_GROUP_CONFIG];
+
+extern esp_ble_mesh_prov_t prov;
+
+extern esp_ble_mesh_model_pub_t model_pub_config;
+
+extern esp_ble_mesh_model_pub_t vendor_model_pub_config;
+
+// configure server module
+extern esp_ble_mesh_cfg_srv_t cfg_srv;
+
+extern esp_ble_mesh_model_t config_server_models[];
+
+extern esp_ble_mesh_elem_t config_server_elements[];
+
+extern esp_ble_mesh_comp_t config_server_comp;
+
+// config client model
+esp_ble_mesh_client_t cfg_cli;
+extern esp_ble_mesh_model_t config_client_models[];
+
+extern esp_ble_mesh_elem_t config_client_elements[];
+
+extern esp_ble_mesh_comp_t config_client_comp;
+
+// configure special module
+extern esp_ble_mesh_model_op_t gen_onoff_srv_model_op_config[];
+
+extern esp_ble_mesh_model_t gen_onoff_srv_models[];
+
+extern esp_ble_mesh_elem_t gen_onoff_srv_elements[];
+
+extern esp_ble_mesh_comp_t gen_onoff_srv_comp;
+
+// config generic onoff client
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+
+extern esp_ble_mesh_client_t gen_onoff_cli;
+
+extern esp_ble_mesh_model_t gen_onoff_cli_models[];
+
+extern esp_ble_mesh_elem_t gen_onoff_cli_elements[];
+
+extern esp_ble_mesh_comp_t gen_onoff_cli_comp;
+#endif //CONFIG_BT_MESH_GENERIC_ONOFF_CLI
+
+//CONFIG VENDOR MODEL TEST PERFORMANCE
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV 0x2000
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI 0x2001
+
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET ESP_BLE_MESH_MODEL_OP_3(0x01, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET ESP_BLE_MESH_MODEL_OP_3(0x02, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK ESP_BLE_MESH_MODEL_OP_3(0x03, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS ESP_BLE_MESH_MODEL_OP_3(0x04, CID_ESP)
+
+extern esp_ble_mesh_client_t test_perf_cli;
+
+extern esp_ble_mesh_model_op_t test_perf_srv_op[];
+
+extern esp_ble_mesh_model_op_t test_perf_cli_op[];
+
+extern esp_ble_mesh_model_t config_models[];
+
+extern esp_ble_mesh_model_t test_perf_cli_models[];
+
+extern esp_ble_mesh_elem_t test_perf_cli_elements[];
+
+extern esp_ble_mesh_comp_t test_perf_cli_comp;
+
+extern esp_ble_mesh_model_t test_perf_srv_models[];
+
+extern esp_ble_mesh_elem_t test_perf_srv_elements[];
+
+extern esp_ble_mesh_comp_t test_perf_srv_comp;
+
+#endif //_BLE_MESH_CFG_SRV_MODEL_H_
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_decl.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_decl.h
new file mode 100644
index 0000000000..bf10edb741
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_decl.h
@@ -0,0 +1,28 @@
+/* Console example — declarations of command registration functions.
+
+ This example code is in the Public Domain (or CC0 licensed, at your option.)
+
+ Unless required by applicable law or agreed to in writing, this
+ software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
+ CONDITIONS OF ANY KIND, either express or implied.
+*/
+#pragma once
+
+#include "esp_ble_mesh_defs.h"
+
+// Register system functions
+void register_system();
+
+// Register blutooth
+void register_bluetooth();
+
+// Register mesh node cmd
+void ble_mesh_register_mesh_node();
+
+// Register mesh config server and generic server operation cmd
+void ble_mesh_register_server();
+
+#if (CONFIG_BT_MESH_CFG_CLI)
+// Register mesh config client operation cmd
+void ble_mesh_register_configuration_client_model();
+#endif
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_lib.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_lib.c
new file mode 100644
index 0000000000..88b728bbab
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_lib.c
@@ -0,0 +1,128 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "ble_mesh_console_lib.h"
+
+static int hex2num(char c);
+static int hex2byte(const char *hex);
+
+static int hex2num(char c)
+{
+ if (c >= '0' && c <= '9') {
+ return c - '0';
+ }
+ if (c >= 'a' && c <= 'f') {
+ return c - 'a' + 10;
+ }
+ if (c >= 'A' && c <= 'F') {
+ return c - 'A' + 10;
+ }
+ return -1;
+}
+
+static int hex2byte(const char *hex)
+{
+ int a, b;
+ a = hex2num(*hex++);
+ if (a < 0) {
+ return -1;
+ }
+ b = hex2num(*hex++);
+ if (b < 0) {
+ return -1;
+ }
+ return (a << 4) | b;
+}
+
+int hexstr_2_bin(const char *hex, uint8_t *buf, uint32_t len)
+{
+ uint32_t i;
+ int a;
+ const char *ipos = hex;
+ uint8_t *opos = buf;
+
+ for (i = 0; i < len; i++) {
+ a = hex2byte(ipos);
+ if (a < 0) {
+ return -1;
+ }
+ *opos ++ = a;
+ ipos += 2;
+ }
+ return 0;
+}
+
+int get_value_string(char *value_in, char *buf)
+{
+ int result = -1;
+
+ uint16_t length = strlen(value_in);
+ for(int i = 0; i 2) {
+ if (value_in[0] == '0' && value_in[1] == 'x') {
+ buf[(length - 2) / 2] = 0;
+ result = hexstr_2_bin(&value_in[2], (uint8_t *)buf, (length - 2) / 2);
+ length = (length - 2) / 2;
+ } else {
+ strcpy(buf, value_in);
+ result = 0;
+ }
+ } else {
+ strcpy(buf, value_in);
+ result = 0;
+ }
+
+ return result;
+}
+
+bool str_2_mac(uint8_t *str, uint8_t *dest)
+{
+ uint8_t loop = 0;
+ uint8_t tmp = 0;
+ uint8_t *src_p = str;
+
+ if (strlen((char *)src_p) != 17) { // must be like 12:34:56:78:90:AB
+ return false;
+ }
+
+ for (loop = 0; loop < 17 ; loop++) {
+ if (loop % 3 == 2) {
+ if (src_p[loop] != ':') {
+ return false;
+ }
+
+ continue;
+ }
+
+ if ((src_p[loop] >= '0') && (src_p[loop] <= '9')) {
+ tmp = tmp * 16 + src_p[loop] - '0';
+ } else if ((src_p[loop] >= 'A') && (src_p[loop] <= 'F')) {
+ tmp = tmp * 16 + src_p[loop] - 'A' + 10;
+ } else if ((src_p[loop] >= 'a') && (src_p[loop] <= 'f')) {
+ tmp = tmp * 16 + src_p[loop] - 'a' + 10;
+ } else {
+ return false;
+ }
+
+ if (loop % 3 == 1) {
+ *dest++ = tmp;
+ tmp = 0;
+ }
+ }
+
+ return true;
+}
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_lib.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_lib.h
new file mode 100644
index 0000000000..8f8449eca4
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_lib.h
@@ -0,0 +1,31 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BLE_MESH_CONSOLE_LIB_H_
+#define _BLE_MESH_CONSOLE_LIB_H_
+
+#include
+#include
+
+#include "esp_system.h"
+#include "esp_log.h"
+#include "nvs_flash.h"
+#include "esp_console.h"
+#include "argtable3/argtable3.h"
+
+bool str_2_mac(uint8_t *str, uint8_t *dest);
+int hexstr_2_bin(const char *hex, uint8_t *buf, uint32_t len);
+int get_value_string(char *value_in, char *buf);
+
+#endif //_BLE_MESH_CONSOLE_LIB_H_
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_main.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_main.c
new file mode 100644
index 0000000000..e629cf3054
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_main.c
@@ -0,0 +1,215 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "esp_system.h"
+#include "esp_log.h"
+#include "esp_vfs_dev.h"
+#include "driver/uart.h"
+#include "nvs.h"
+#include "nvs_flash.h"
+
+#include "esp_bt.h"
+#include "esp_bt_main.h"
+
+#include "esp_console.h"
+#include "linenoise/linenoise.h"
+#include "argtable3/argtable3.h"
+
+#include "ble_mesh_console_decl.h"
+
+#if CONFIG_STORE_HISTORY
+
+#define MOUNT_PATH "/data"
+#define HISTORY_PATH MOUNT_PATH "/history.txt"
+
+static void initialize_filesystem()
+{
+ static wl_handle_t wl_handle;
+ const esp_vfs_fat_mount_config_t mount_config = {
+ .max_files = 4,
+ .format_if_mount_failed = true
+ };
+ esp_err_t err = esp_vfs_fat_spiflash_mount(MOUNT_PATH, "storage", &mount_config, &wl_handle);
+ if (err != ESP_OK) {
+ printf("Failed to mount FATFS (0x%x)", err);
+ return;
+ }
+}
+#endif // CONFIG_STORE_HISTORY
+
+static void initialize_console()
+{
+ /* Disable buffering on stdin and stdout */
+ setvbuf(stdin, NULL, _IONBF, 0);
+ setvbuf(stdout, NULL, _IONBF, 0);
+
+ /* Minicom, screen, idf_monitor send CR when ENTER key is pressed */
+ esp_vfs_dev_uart_set_rx_line_endings(ESP_LINE_ENDINGS_CR);
+ /* Move the caret to the beginning of the next line on '\n' */
+ esp_vfs_dev_uart_set_tx_line_endings(ESP_LINE_ENDINGS_CRLF);
+
+ /* Install UART driver for interrupt-driven reads and writes */
+ ESP_ERROR_CHECK( uart_driver_install(CONFIG_CONSOLE_UART_NUM,
+ 256, 0, 0, NULL, 0) );
+
+ /* Tell VFS to use UART driver */
+ esp_vfs_dev_uart_use_driver(CONFIG_CONSOLE_UART_NUM);
+
+ /* Initialize the console */
+ esp_console_config_t console_config = {
+ .max_cmdline_args = 20,
+ .max_cmdline_length = 256,
+#if CONFIG_LOG_COLORS
+ .hint_color = atoi(LOG_COLOR_CYAN)
+#endif
+ };
+ ESP_ERROR_CHECK( esp_console_init(&console_config) );
+
+ /* Configure linenoise line completion library */
+ /* Enable multiline editing. If not set, long commands will scroll within
+ * single line.
+ */
+ linenoiseSetMultiLine(1);
+
+ /* Tell linenoise where to get command completions and hints */
+ linenoiseSetCompletionCallback(&esp_console_get_completion);
+ linenoiseSetHintsCallback((linenoiseHintsCallback *) &esp_console_get_hint);
+
+ /* Set command history size */
+ linenoiseHistorySetMaxLen(100);
+
+#if CONFIG_STORE_HISTORY
+ /* Load command history from filesystem */
+ linenoiseHistoryLoad(HISTORY_PATH);
+#endif
+}
+
+
+esp_err_t bluetooth_init(void)
+{
+ esp_err_t ret;
+
+ esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
+ ret = esp_bt_controller_init(&bt_cfg);
+ if (ret) {
+ printf("%s initialize controller failed\n", __func__);
+ return ret;
+ }
+
+ ret = esp_bt_controller_enable(ESP_BT_MODE_BLE);
+ if (ret) {
+ printf("%s enable controller failed\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_init();
+ if (ret) {
+ printf("%s init bluetooth failed\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_enable();
+ if (ret) {
+ printf("%s enable bluetooth failed\n", __func__);
+ return ret;
+ }
+
+ return ret;
+}
+
+void app_main(void)
+{
+ esp_err_t res;
+
+ nvs_flash_init();
+
+ // init and enable bluetooth
+ res = bluetooth_init();
+ if (res) {
+ printf("esp32_bluetooth_init failed (ret %d)", res);
+ }
+
+#if CONFIG_STORE_HISTORY
+ initialize_filesystem();
+#endif
+
+ initialize_console();
+
+ /* Register commands */
+ esp_console_register_help_command();
+ register_system();
+ register_bluetooth();
+ ble_mesh_register_mesh_node();
+ ble_mesh_register_server();
+
+ /* Prompt to be printed before each line.
+ * This can be customized, made dynamic, etc.
+ */
+ const char *prompt = LOG_COLOR_I "esp32> " LOG_RESET_COLOR;
+
+ printf("\n"
+ "This is an example of ESP-IDF console component.\n"
+ "Type 'help' to get the list of commands.\n"
+ "Use UP/DOWN arrows to navigate through command history.\n"
+ "Press TAB when typing command name to auto-complete.\n");
+
+ /* Figure out if the terminal supports escape sequences */
+ int probe_status = linenoiseProbe();
+ if (probe_status) { /* zero indicates success */
+ printf("\n"
+ "Your terminal application does not support escape sequences.\n"
+ "Line editing and history features are disabled.\n"
+ "On Windows, try using Putty instead.\n");
+ linenoiseSetDumbMode(1);
+#if CONFIG_LOG_COLORS
+ /* Since the terminal doesn't support escape sequences,
+ * don't use color codes in the prompt.
+ */
+ prompt = "esp32> ";
+#endif //CONFIG_LOG_COLORS
+ }
+
+ /* Main loop */
+ while (true) {
+ /* Get a line using linenoise.
+ * The line is returned when ENTER is pressed.
+ */
+ char *line = linenoise(prompt);
+ if (line == NULL) { /* Ignore empty lines */
+ continue;
+ }
+ /* Add the command to the history */
+ linenoiseHistoryAdd(line);
+#if CONFIG_STORE_HISTORY
+ /* Save command history to filesystem */
+ linenoiseHistorySave(HISTORY_PATH);
+#endif
+
+ /* Try to run the command */
+ int ret;
+ esp_err_t err = esp_console_run(line, &ret);
+ if (err == ESP_ERR_NOT_FOUND) {
+ printf("Unrecognized command\n");
+ } else if (err == ESP_ERR_INVALID_ARG) {
+ // command was empty
+ } else if (err == ESP_OK && ret != ESP_OK) {
+ printf("\nCommand returned non-zero error code: 0x%x\n", ret);
+ } else if (err != ESP_OK) {
+ printf("Internal error: 0x%x\n", err);
+ }
+ /* linenoise allocates line buffer on the heap, so need to free it */
+ linenoiseFree(line);
+ }
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_system.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_system.c
new file mode 100644
index 0000000000..ebb7d83e29
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_console_system.c
@@ -0,0 +1,179 @@
+/* Console example — various system commands
+
+ This example code is in the Public Domain (or CC0 licensed, at your option.)
+
+ Unless required by applicable law or agreed to in writing, this
+ software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
+ CONDITIONS OF ANY KIND, either express or implied.
+*/
+
+#include
+#include
+#include
+
+#include "esp_log.h"
+#include "esp_console.h"
+#include "esp_system.h"
+#include "esp_sleep.h"
+#include "driver/rtc_io.h"
+#include "soc/rtc_cntl_reg.h"
+#include "argtable3/argtable3.h"
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+
+#include "ble_mesh_console_decl.h"
+
+static void register_free();
+static void register_restart();
+static void register_make();
+
+void register_system()
+{
+ register_free();
+ register_restart();
+ register_make();
+}
+
+/** 'restart' command restarts the program */
+
+static int restart(int argc, char **argv)
+{
+ printf("%s, %s", __func__, "Restarting");
+ esp_restart();
+}
+
+static void register_restart()
+{
+ const esp_console_cmd_t cmd = {
+ .command = "restart",
+ .help = "Restart the program",
+ .hint = NULL,
+ .func = &restart,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) );
+}
+
+/** 'free' command prints available heap memory */
+
+static int free_mem(int argc, char **argv)
+{
+ printf("%d\n", esp_get_free_heap_size());
+ return 0;
+}
+
+static void register_free()
+{
+ const esp_console_cmd_t cmd = {
+ .command = "free",
+ .help = "Get the total size of heap memory available",
+ .hint = NULL,
+ .func = &free_mem,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) );
+}
+
+static int make(int argc, char **argv)
+{
+ int count = REG_READ(RTC_CNTL_STORE0_REG);
+ if (++count >= 3) {
+ printf("This is not the console you are looking for.\n");
+ return 0;
+ }
+ REG_WRITE(RTC_CNTL_STORE0_REG, count);
+
+ const char *make_output =
+ R"(LD build/console.elf
+esptool.py v2.1-beta1
+)";
+
+ const char* flash_output[] = {
+R"(Flashing binaries to serial port )" CONFIG_ESPTOOLPY_PORT R"( (app at offset 0x10000)...
+esptool.py v2.1-beta1
+Connecting....
+)",
+R"(Chip is ESP32D0WDQ6 (revision 0)
+Uploading stub...
+Running stub...
+Stub running...
+Changing baud rate to 921600
+Changed.
+Configuring flash size...
+Auto-detected Flash size: 4MB
+Flash params set to 0x0220
+Compressed 15712 bytes to 9345...
+)",
+R"(Wrote 15712 bytes (9345 compressed) at 0x00001000 in 0.1 seconds (effective 1126.9 kbit/s)...
+Hash of data verified.
+Compressed 333776 bytes to 197830...
+)",
+R"(Wrote 333776 bytes (197830 compressed) at 0x00010000 in 3.3 seconds (effective 810.3 kbit/s)...
+Hash of data verified.
+Compressed 3072 bytes to 82...
+)",
+R"(Wrote 3072 bytes (82 compressed) at 0x00008000 in 0.0 seconds (effective 1588.4 kbit/s)...
+Hash of data verified.
+Leaving...
+Hard resetting...
+)"
+ };
+
+ const char* monitor_output =
+R"(MONITOR
+)" LOG_COLOR_W R"(--- idf_monitor on )" CONFIG_ESPTOOLPY_PORT R"( 115200 ---
+--- Quit: Ctrl+] | Menu: Ctrl+T | Help: Ctrl+T followed by Ctrl+H --
+)" LOG_RESET_COLOR;
+
+ bool need_make = false;
+ bool need_flash = false;
+ bool need_monitor = false;
+ for (int i = 1; i < argc; ++i) {
+ if (strcmp(argv[i], "all") == 0) {
+ need_make = true;
+ } else if (strcmp(argv[i], "flash") == 0) {
+ need_make = true;
+ need_flash = true;
+ } else if (strcmp(argv[i], "monitor") == 0) {
+ need_monitor = true;
+ } else if (argv[i][0] == '-') {
+ /* probably -j option */
+ } else if (isdigit((int) argv[i][0])) {
+ /* might be an argument to -j */
+ } else {
+ printf("make: *** No rule to make target `%s'. Stop.\n", argv[i]);
+ /* Technically this is an error, but let's not spoil the output */
+ return 0;
+ }
+ }
+ if (argc == 1) {
+ need_make = true;
+ }
+ if (need_make) {
+ printf("%s", make_output);
+ }
+ if (need_flash) {
+ size_t n_items = sizeof(flash_output) / sizeof(flash_output[0]);
+ for (int i = 0; i < n_items; ++i) {
+ printf("%s", flash_output[i]);
+ vTaskDelay(200/portTICK_PERIOD_MS);
+ }
+ }
+ if (need_monitor) {
+ printf("%s", monitor_output);
+ esp_restart();
+ }
+ return 0;
+}
+
+static void register_make()
+{
+ const esp_console_cmd_t cmd = {
+ .command = "make",
+ .help = NULL, /* Do not include in 'help' output */
+ .hint = "all | flash | monitor",
+ .func = &make,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) );
+}
+
+
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_register_node_cmd.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_register_node_cmd.c
new file mode 100644
index 0000000000..cf181a6ccb
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_register_node_cmd.c
@@ -0,0 +1,547 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "esp_bt.h"
+#include "soc/soc.h"
+
+#include "esp_bt_device.h"
+
+#include "test.h"
+#include "esp_ble_mesh_networking_api.h"
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_common_api.h"
+#include "esp_ble_mesh_provisioning_api.h"
+
+#include "ble_mesh_console_lib.h"
+#include "ble_mesh_adapter.h"
+
+typedef struct {
+ struct arg_str *static_val;
+ struct arg_int *static_val_len;
+ struct arg_int *output_size;
+ struct arg_int *output_actions;
+ struct arg_int *input_size;
+ struct arg_int *input_actions;
+ struct arg_end *end;
+} ble_mesh_prov_t;
+static ble_mesh_prov_t oob;
+
+typedef struct {
+ struct arg_int *model_type;
+ struct arg_int *config_index;
+ struct arg_str *dev_uuid;
+ struct arg_int *pub_config;
+ struct arg_end *end;
+} ble_mesh_comp_t;
+static ble_mesh_comp_t component;
+
+typedef struct {
+ struct arg_int *bearer;
+ struct arg_int *enable;
+ struct arg_end *end;
+} ble_mesh_bearer_t;
+static ble_mesh_bearer_t bearer;
+
+typedef struct {
+ struct arg_str *action_type;
+ struct arg_int *package_num;
+ struct arg_end *end;
+} ble_mesh_node_statistices_t;
+ble_mesh_node_statistices_t node_statistices;
+
+typedef struct {
+ struct arg_str *action_type;
+ struct arg_int *tx_sense_power;
+ struct arg_end *end;
+} ble_mesh_tx_sense_power;
+static ble_mesh_tx_sense_power power_set;
+
+typedef struct {
+ struct arg_str *net_key;
+ struct arg_int *net_idx;
+ struct arg_int *unicast_addr;
+ struct arg_str *dev_key;
+ struct arg_str *app_key;
+ struct arg_int *app_idx;
+ struct arg_int *group_addr;
+ struct arg_end *end;
+} ble_mesh_node_network_info_t;
+ble_mesh_node_network_info_t node_network_info;
+
+ble_mesh_node_status node_status = {
+ .previous = 0x0,
+ .current = 0x0,
+};
+
+SemaphoreHandle_t ble_mesh_node_sema;
+
+void ble_mesh_register_node_cmd();
+// Register callback function
+void ble_mesh_prov_cb(esp_ble_mesh_prov_cb_event_t event, esp_ble_mesh_prov_cb_param_t *param);
+void ble_mesh_model_cb(esp_ble_mesh_model_cb_event_t event, esp_ble_mesh_model_cb_param_t *param);
+
+
+void ble_mesh_register_mesh_node()
+{
+ ble_mesh_register_node_cmd();
+}
+
+int ble_mesh_register_node_cb()
+{
+ ESP_LOGD(TAG, "enter %s\n", __func__);
+ ble_mesh_node_init();
+ esp_ble_mesh_register_prov_callback(ble_mesh_prov_cb);
+ esp_ble_mesh_register_custom_model_callback(ble_mesh_model_cb);
+ ESP_LOGI(TAG, "Node:Reg,OK");
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return 0;
+}
+
+void ble_mesh_prov_cb(esp_ble_mesh_prov_cb_event_t event, esp_ble_mesh_prov_cb_param_t *param)
+{
+ ESP_LOGD(TAG, "enter %s, event = %d", __func__, event);
+ switch (event) {
+ case ESP_BLE_MESH_PROV_REGISTER_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->prov_register_comp.err_code, "Provisioning:Register");
+ break;
+ case ESP_BLE_MESH_NODE_PROV_ENABLE_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_prov_enable_comp.err_code, "Node:EnBearer");
+ break;
+ case ESP_BLE_MESH_NODE_PROV_DISABLE_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_prov_disable_comp.err_code, "Node:DisBearer");
+ break;
+ case ESP_BLE_MESH_NODE_PROV_LINK_OPEN_EVT:
+ ESP_LOGI(TAG, "Node:LinkOpen,OK,%d", param->node_prov_link_open.bearer);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_LINK_CLOSE_EVT:
+ ESP_LOGI(TAG, "Node:LinkClose,OK,%d", param->node_prov_link_close.bearer);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_NUMBER_EVT:
+ ESP_LOGI(TAG, "Node:OutPut,%d,%d", param->node_prov_output_num.action, param->node_prov_output_num.number);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_OUTPUT_STRING_EVT:
+ ESP_LOGI(TAG, "Node:OutPutStr,%s", param->node_prov_output_str.string);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_INPUT_EVT:
+ ESP_LOGI(TAG, "Node:InPut,%d,%d", param->node_prov_input.action, param->node_prov_input.size);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_COMPLETE_EVT:
+ ESP_LOGI(TAG, "Provisioning:Success,%d", param->node_prov_complete.addr);
+ ble_mesh_set_node_prestore_params(param->node_prov_complete.net_idx, param->node_prov_complete.addr);
+ break;
+ case ESP_BLE_MESH_NODE_PROV_RESET_EVT:
+ ESP_LOGI(TAG, "Node:Reset");
+ break;
+ case ESP_BLE_MESH_NODE_PROV_INPUT_NUMBER_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_prov_input_num_comp.err_code, "Node:InputNum");
+ break;
+ case ESP_BLE_MESH_NODE_PROV_INPUT_STRING_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_prov_input_str_comp.err_code, "Node:InputStr");
+ break;
+ case ESP_BLE_MESH_NODE_SET_UNPROV_DEV_NAME_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_set_unprov_dev_name_comp.err_code, "Node:SetName");
+ break;
+ case ESP_BLE_MESH_NODE_PROXY_IDENTITY_ENABLE_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_proxy_identity_enable_comp.err_code, "Node:ProxyIndentity");
+ break;
+ case ESP_BLE_MESH_NODE_PROXY_GATT_ENABLE_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_proxy_gatt_enable_comp.err_code, "Node:EnProxyGatt");
+ break;
+ case ESP_BLE_MESH_NODE_PROXY_GATT_DISABLE_COMP_EVT:
+ ble_mesh_callback_check_err_code(param->node_proxy_gatt_disable_comp.err_code, "Node:DisProxyGatt");
+ break;
+ default:
+ break;
+ }
+ ESP_LOGD(TAG, "exit %s", __func__);
+}
+
+void ble_mesh_model_cb(esp_ble_mesh_model_cb_event_t event, esp_ble_mesh_model_cb_param_t *param)
+{
+ uint8_t status;
+ uint16_t result;
+ uint8_t data[4];
+
+ ESP_LOGD(TAG, "enter %s, event=%x\n", __func__, event);
+ printf("enter %s, event=%x\n", __func__, event);
+ switch (event) {
+ case ESP_BLE_MESH_MODEL_OPERATION_EVT:
+ if (param->model_operation.model != NULL && param->model_operation.model->op != NULL) {
+ if (param->model_operation.opcode == BT_MESH_MODEL_OP_GEN_ONOFF_GET) {
+ ESP_LOGI(TAG, "Node:GetStatus,Success");
+ ble_mesh_node_get_state(status);
+ esp_ble_mesh_server_model_send_msg(param->model_operation.model, param->model_operation.ctx, ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS,
+ sizeof(status), &status);
+ } else if (param->model_operation.opcode == ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET) {
+ ESP_LOGI(TAG, "Node:SetAck,Success,%d,%d,%d", param->model_operation.msg[0], param->model_operation.ctx->recv_ttl, param->model_operation.length);
+ ble_mesh_node_set_state(param->model_operation.msg[0]);
+ ble_mesh_node_get_state(status);
+ esp_ble_mesh_server_model_send_msg(param->model_operation.model, param->model_operation.ctx, ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS,
+ sizeof(status), &status);
+ } else if (param->model_operation.opcode == ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_SET_UNACK) {
+ ble_mesh_node_set_state(param->model_operation.msg[0]);
+ ESP_LOGI(TAG, "Node:SetUnAck,Success,%d,%d", param->model_operation.msg[0], param->model_operation.ctx->recv_ttl);
+ } else if (param->model_operation.opcode == ESP_BLE_MESH_MODEL_OP_GEN_ONOFF_STATUS) {
+ ESP_LOGI(TAG, "Node:Status,Success,%d", param->model_operation.length);
+ } else if (param->model_operation.opcode == ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET) {
+ ESP_LOGI(TAG, "VendorModel:SetAck,Success,%d", param->model_operation.ctx->recv_ttl);
+ data[0] = param->model_operation.msg[0];
+ data[1] = param->model_operation.msg[1];
+ data[2] = param->model_operation.msg[2];
+ data[3] = param->model_operation.ctx->recv_ttl;
+ result = ble_mesh_node_statistics_accumultate(param->model_operation.msg, param->model_operation.length, VENDOR_MODEL_PERF_OPERATION_TYPE_SET);
+ if (result == 0) {
+ esp_ble_mesh_server_model_send_msg(param->model_operation.model, param->model_operation.ctx,
+ ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS, sizeof(data), data);
+ }
+ } else if (param->model_operation.opcode == ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK) {
+ ESP_LOGI(TAG, "VendorModel:SetUnAck,Success,%d,%d", param->model_operation.ctx->recv_ttl, param->model_operation.length);
+ result = ble_mesh_node_statistics_accumultate(param->model_operation.msg, param->model_operation.length, VENDOR_MODEL_PERF_OPERATION_TYPE_SET_UNACK);
+ if (result == 0) {
+ esp_ble_mesh_server_model_send_msg(param->model_operation.model, param->model_operation.ctx,
+ ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS, param->model_operation.length, param->model_operation.msg);
+ }
+ }
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_SEND_COMP_EVT:
+ if (param->model_send_comp.err_code == ESP_OK) {
+ ESP_LOGI(TAG, "Node:ModelSend,OK");
+ } else {
+ ESP_LOGE(TAG, "Node:ModelSend,Fail");
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_PUBLISH_COMP_EVT:
+ ESP_LOGI(TAG, "PublishSend,OK,0x%x,%d,", param->model_publish_comp.model->model_id, param->model_publish_comp.model->pub->msg->len);
+ break;
+ case ESP_BLE_MESH_MODEL_PUBLISH_UPDATE_EVT:
+ ESP_LOGI(TAG, "PublishUpdate,OK");
+ break;
+ case ESP_BLE_MESH_CLIENT_MODEL_SEND_TIMEOUT_EVT:
+ ESP_LOGI(TAG, "Node:TimeOut");
+ break;
+ case ESP_BLE_MESH_MODEL_EVT_MAX:
+ ESP_LOGI(TAG, "Node:MaxEvt");
+ break;
+ default:
+ break;
+ }
+
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+}
+
+int ble_mesh_power_set(int argc, char **argv)
+{
+ esp_err_t result = ESP_OK;
+ ESP_LOGD(TAG, "enter %s\n", __func__);
+ int nerrors = arg_parse(argc, argv, (void **) &power_set);
+ if (nerrors != 0) {
+ arg_print_errors(stderr, power_set.end, argv[0]);
+ return 1;
+ }
+
+ if (strcmp(power_set.action_type->sval[0], "tx") == 0) {
+ result = esp_ble_tx_power_set(ESP_BLE_PWR_TYPE_ADV, power_set.tx_sense_power->ival[0]);
+ } else if (strcmp(power_set.action_type->sval[0], "sense") == 0) {
+ uint32_t *reg = (uint32_t *)(0x6001c07c);
+ int reg_addr = 0x6001c07c;
+ uint32_t flag = 0x00FF0000;
+ uint32_t sense_new = power_set.tx_sense_power->ival[0];
+ uint32_t reg_to_write = ((*reg) &= ~flag) | ((256 - sense_new) << 16);
+ REG_WRITE(reg_addr, reg_to_write);
+
+ }
+
+ if (result == ESP_OK) {
+ ESP_LOGI(TAG, "Node:SetPower,OK\n");
+ }
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return result;
+}
+
+static int ble_mesh_load_oob(int argc, char **argv)
+{
+ uint8_t *static_val;
+ int nerrors = arg_parse(argc, argv, (void **) &oob);
+
+ ESP_LOGD(TAG, "enter %s \n", __func__);
+
+ if (nerrors != 0) {
+ arg_print_errors(stderr, oob.end, argv[0]);
+ return 1;
+ }
+
+ //parsing prov
+ if (oob.static_val->count != 0) {
+ static_val = malloc(oob.static_val_len->ival[0] + 1);
+ get_value_string((char *)oob.static_val->sval[0], (char *)static_val);
+ prov.static_val = static_val;
+ }
+
+ arg_int_to_value(oob.static_val_len, prov.static_val_len, "static_val_len");
+ arg_int_to_value(oob.output_size, prov.output_size, "output_size");
+ arg_int_to_value(oob.output_actions, prov.output_actions, "output_actions");
+ arg_int_to_value(oob.input_size, prov.input_size, "input_size");
+ arg_int_to_value(oob.input_actions, prov.input_actions, "input_action");
+
+ ESP_LOGI(TAG, "OOB:Load,OK\n");
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return 0;
+}
+
+int ble_mesh_init(int argc, char **argv)
+{
+ int err;
+ esp_ble_mesh_comp_t *local_component = NULL;
+ uint8_t *device_uuid =NULL;
+
+ int nerrors = arg_parse(argc, argv, (void **) &component);
+ if (nerrors != 0) {
+ arg_print_errors(stderr, component.end, argv[0]);
+ return 1;
+ }
+
+ ESP_LOGD(TAG, "enter %s, module %x\n", __func__, component.model_type->ival[0]);
+ local_component = ble_mesh_get_component(component.model_type->ival[0]);
+
+ if (component.dev_uuid->count != 0) {
+ device_uuid = malloc((16 + 1) * sizeof(uint8_t));
+ if (device_uuid == NULL) {
+ ESP_LOGE(TAG, "ble mesh malloc failed, %d\n", __LINE__);
+ }
+ get_value_string((char *)component.dev_uuid->sval[0], (char *) device_uuid);
+ memcpy(dev_uuid, device_uuid, 16);
+ } else {
+ memcpy(dev_uuid, esp_bt_dev_get_address(), 6);
+ }
+
+ err = esp_ble_mesh_init(&prov, local_component);
+ if (err) {
+ ESP_LOGE(TAG, "Initializing mesh failed (err %d)\n", err);
+ return err;
+ }
+
+ free(device_uuid);
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return err;
+}
+
+int ble_mesh_node_enable_bearer(int argc, char **argv)
+{
+ esp_err_t err = 0;
+
+ ESP_LOGD(TAG, "enter %s \n", __func__);
+
+ int nerrors = arg_parse(argc, argv, (void **) &bearer);
+ if (nerrors != 0) {
+ arg_print_errors(stderr, bearer.end, argv[0]);
+ return 1;
+ }
+
+ if (bearer.enable->count != 0) {
+ if (bearer.enable->ival[0]) {
+ //err = esp_ble_tx_power_set(ESP_BLE_PWR_TYPE_ADV, ESP_PWR_LVL_N12);
+ err = esp_ble_mesh_node_prov_enable(bearer.bearer->ival[0]);
+ } else {
+ err = esp_ble_mesh_node_prov_disable(bearer.bearer->ival[0]);
+ }
+ } else {
+ return 1;
+ }
+
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return err;
+}
+
+int ble_mesh_node_reset()
+{
+ esp_err_t err;
+ ESP_LOGD(TAG, "enter %s\n", __func__);
+
+ err = esp_ble_mesh_node_local_reset();
+
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return err;
+}
+
+int ble_mesh_node_statistics_regist(int argc, char **argv)
+{
+ int result = ESP_OK;
+
+ int nerrors = arg_parse(argc, argv, (void **) &node_statistices);
+ if (nerrors != 0) {
+ arg_print_errors(stderr, node_statistices.end, argv[0]);
+ return 1;
+ }
+
+ ESP_LOGD(TAG, "enter %s\n", __func__);
+
+ if (strcmp(node_statistices.action_type->sval[0], "init") == 0) {
+ result = ble_mesh_node_statistics_init(node_statistices.package_num->ival[0]);
+ ESP_LOGI(TAG, "Node:InitStatistics,OK\n");
+ } else if (strcmp(node_statistices.action_type->sval[0], "get") == 0) {
+ ble_mesh_node_statistics_get();
+ ESP_LOGI(TAG, "Node:GetStatistics,OK\n");
+ } else if (strcmp(node_statistices.action_type->sval[0], "destroy") == 0) {
+ ble_mesh_node_statistics_destroy();
+ ESP_LOGI(TAG, "Node:DestroyStatistics\n");
+ }
+
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return result;
+}
+
+int ble_mesh_node_enter_network_auto(int argc, char **argv)
+{
+ esp_err_t err = ESP_OK;
+ struct bt_mesh_device_network_info info = {
+ .flags = 0,
+ .iv_index = 0,
+ };
+
+ int nerrors = arg_parse(argc, argv, (void **) &node_network_info);
+ if (nerrors != 0) {
+ arg_print_errors(stderr, node_network_info.end, argv[0]);
+ return 1;
+ }
+
+ ESP_LOGD(TAG, "enter %s\n", __func__);
+
+ arg_int_to_value(node_network_info.net_idx, info.net_idx, "network key index");
+ arg_int_to_value(node_network_info.unicast_addr, info.unicast_addr, "unicast address");
+ arg_int_to_value(node_network_info.app_idx, info.app_idx, "appkey index");
+ arg_int_to_value(node_network_info.group_addr, info.group_addr, "group address");
+ err = get_value_string((char *)node_network_info.net_key->sval[0], (char *)info.net_key);
+ err = get_value_string((char *)node_network_info.dev_key->sval[0], (char *)info.dev_key);
+ err = get_value_string((char *)node_network_info.app_key->sval[0], (char *)info.app_key);
+
+ err = bt_mesh_device_auto_enter_network(&info);
+ if (err == ESP_OK) {
+ ESP_LOGD(TAG, "NODE:EnNetwork,OK");
+ } else {
+ ESP_LOGE(TAG, "NODE:EnNetwork,FAIL,%d", err);
+ }
+
+ ESP_LOGD(TAG, "exit %s\n", __func__);
+ return err;
+}
+
+void ble_mesh_register_node_cmd()
+{
+ const esp_console_cmd_t register_cmd = {
+ .command = "bmreg",
+ .help = "ble mesh: provisioner/node register callback",
+ .hint = NULL,
+ .func = &ble_mesh_register_node_cb,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(®ister_cmd));
+
+ oob.static_val = arg_str0("s", NULL, "", "Static OOB value");
+ oob.static_val_len = arg_int0("l", NULL, "", "Static OOB value length");
+ oob.output_size = arg_int0("x", NULL, "", "Maximum size of Output OOB");
+ oob.output_actions = arg_int0("o", NULL, "", "Supported Output OOB Actions");
+ oob.input_size = arg_int0("y", NULL, "", "Maximum size of Input OOB");
+ oob.input_actions = arg_int0("i", NULL, "", "Supported Input OOB Actions");
+ oob.end = arg_end(1);
+
+ const esp_console_cmd_t oob_cmd = {
+ .command = "bmoob",
+ .help = "ble mesh: provisioner/node config OOB parameters",
+ .hint = NULL,
+ .func = &ble_mesh_load_oob,
+ .argtable = &oob,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&oob_cmd) );
+
+ component.model_type = arg_int0("m", NULL, "", "mesh model");
+ component.config_index = arg_int0("c", NULL, "", "mesh model op");
+ component.config_index->ival[0] = 0; // set default value
+ component.pub_config = arg_int0("p", NULL, "", "publish message buffer");
+ component.dev_uuid = arg_str0("d", NULL, "", "device uuid");
+ component.end = arg_end(1);
+
+ const esp_console_cmd_t model_cmd = {
+ .command = "bminit",
+ .help = "ble mesh: provisioner/node init",
+ .hint = NULL,
+ .func = &ble_mesh_init,
+ .argtable = &component,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&model_cmd) );
+
+ bearer.bearer = arg_int0("b", NULL, "", "supported bearer");
+ bearer.enable = arg_int0("e", NULL, "", "bearers node supported");
+ bearer.end = arg_end(1);
+
+ const esp_console_cmd_t bearer_cmd = {
+ .command = "bmnbearer",
+ .help = "ble mesh node: enable/disable different bearer",
+ .hint = NULL,
+ .func = &ble_mesh_node_enable_bearer,
+ .argtable = &bearer,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(&bearer_cmd));
+
+ const esp_console_cmd_t reset_cmd = {
+ .command = "bmnreset",
+ .help = "ble mesh node: reset",
+ .hint = NULL,
+ .func = &ble_mesh_node_reset,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(&reset_cmd));
+
+ node_statistices.action_type = arg_str1("z", NULL, "", "action type");
+ node_statistices.package_num = arg_int0("p", NULL, "", "package number");
+ node_statistices.end = arg_end(1);
+
+ const esp_console_cmd_t node_statistices_cmd = {
+ .command = "bmsperf",
+ .help = "ble mesh server: performance statistics",
+ .hint = NULL,
+ .func = &ble_mesh_node_statistics_regist,
+ .argtable = &node_statistices,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(&node_statistices_cmd));
+
+ power_set.action_type = arg_str1("z", NULL, "", "action type");
+ power_set.tx_sense_power = arg_int0("t", NULL, "", "tx power or sense");
+ power_set.end = arg_end(1);
+
+ const esp_console_cmd_t power_set_cmd = {
+ .command = "bmtxpower",
+ .help = "ble mesh: set tx power or sense",
+ .hint = NULL,
+ .func = &ble_mesh_power_set,
+ .argtable = &power_set,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(&power_set_cmd));
+
+ node_network_info.net_key = arg_str1("k", NULL, "", "network key");
+ node_network_info.net_idx = arg_int1("n", NULL, "", "network key index");
+ node_network_info.unicast_addr = arg_int1("u", NULL, "", "unicast address");
+ node_network_info.dev_key = arg_str1("d", NULL, "", "device key");
+ node_network_info.app_key = arg_str1("a", NULL, "", "app key");
+ node_network_info.app_idx = arg_int1("i", NULL, "", "appkey index");
+ node_network_info.group_addr = arg_int1("g", NULL, "", "group address");
+ node_network_info.end = arg_end(1);
+
+ const esp_console_cmd_t node_network_info_cmd = {
+ .command = "bmnnwk",
+ .help = "ble mesh node: auto join network",
+ .hint = NULL,
+ .func = &ble_mesh_node_enter_network_auto,
+ .argtable = &node_network_info,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(&node_network_info_cmd));
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_register_server_cmd.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_register_server_cmd.c
new file mode 100644
index 0000000000..3abe85dc01
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/ble_mesh_register_server_cmd.c
@@ -0,0 +1,83 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_networking_api.h"
+
+#include "ble_mesh_console_lib.h"
+#include "ble_mesh_adapter.h"
+
+void ble_mesh_register_server_operation();
+
+typedef struct {
+ struct arg_str *data;
+ struct arg_int *opcode;
+ struct arg_int *model;
+ struct arg_int *role;
+ struct arg_end *end;
+} ble_mesh_publish_message;
+ble_mesh_publish_message msg_publish;
+
+void ble_mesh_register_server()
+{
+ ble_mesh_register_server_operation();
+}
+
+int ble_mesh_module_publish_message(int argc, char **argv)
+{
+ esp_err_t err;
+ esp_ble_mesh_model_t *model = NULL;
+ uint8_t *data = NULL;
+ uint8_t device_role = ROLE_NODE;
+ uint16_t length = 0;
+
+ ESP_LOGD(TAG, "enter %s \n", __func__);
+
+ int nerrors = arg_parse(argc, argv, (void **) &msg_publish);
+ if (nerrors != 0) {
+ arg_print_errors(stderr, msg_publish.end, argv[0]);
+ return 1;
+ }
+
+ data = malloc(strlen(msg_publish.data->sval[0]));
+ get_value_string((char *)msg_publish.data->sval[0], (char *) data);
+
+ arg_int_to_value(msg_publish.role, device_role, "device role");
+ model = ble_mesh_get_model(msg_publish.model->ival[0]);
+
+ err = esp_ble_mesh_model_publish(model, msg_publish.opcode->ival[0], length, data, device_role);
+
+ ESP_LOGD(TAG, "exit %s \n", __func__);
+ free(data);
+ return err;
+}
+
+void ble_mesh_register_server_operation()
+{
+ msg_publish.data = arg_str1("d", NULL, "", "message data");
+ msg_publish.opcode = arg_int1("o", NULL, "", "operation opcode");
+ msg_publish.model = arg_int1("m", NULL, "", "module published to");
+ msg_publish.role = arg_int1("r", NULL, "", "device role");
+ msg_publish.end = arg_end(1);
+
+ const esp_console_cmd_t msg_publish_cmd = {
+ .command = "bmpublish",
+ .help = "ble mesh: publish message",
+ .hint = NULL,
+ .func = &ble_mesh_module_publish_message,
+ .argtable = &msg_publish,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(&msg_publish_cmd));
+}
+
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/component.mk b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/component.mk
new file mode 100755
index 0000000000..0b9d7585e7
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/component.mk
@@ -0,0 +1,5 @@
+#
+# "main" pseudo-component makefile.
+#
+# (Uses default behaviour of compiling all source files in directory, adding 'include' to include path.)
+
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/register_bluetooth.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/register_bluetooth.c
new file mode 100644
index 0000000000..5afaca813b
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/main/register_bluetooth.c
@@ -0,0 +1,45 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "esp_bt_device.h"
+#include "esp_console.h"
+
+#define MAC2STR(a) (a)[0], (a)[1], (a)[2], (a)[3], (a)[4], (a)[5]
+#define MACSTR "%02x:%02x:%02x:%02x:%02x:%02x"
+
+void register_ble_address();
+
+void register_bluetooth()
+{
+ register_ble_address();
+}
+
+int bt_mac()
+{
+ const uint8_t *mac = esp_bt_dev_get_address();
+ printf("+BTMAC:"MACSTR"\n", MAC2STR(mac));
+ return 0;
+}
+
+void register_ble_address()
+{
+ const esp_console_cmd_t cmd = {
+ .command = "btmac",
+ .help = "get BT mac address",
+ .hint = NULL,
+ .func = (esp_console_cmd_func_t)&bt_mac,
+ };
+ ESP_ERROR_CHECK(esp_console_cmd_register(&cmd));
+}
+
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/sdkconfig.defaults b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/sdkconfig.defaults
new file mode 100644
index 0000000000..1bc06c97de
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_node/sdkconfig.defaults
@@ -0,0 +1,32 @@
+# Override some defaults so BT stack is enabled
+# by default in this example
+CONFIG_CONSOLE_UART_BAUDRATE=921600
+CONFIG_ESPTOOLPY_BAUD_921600B=y
+CONFIG_MONITOR_BAUD_921600B=y
+CONFIG_BT_ENABLED=y
+CONFIG_BT_MESH=y
+CONFIG_BT_MESH_NODE=y
+CONFIG_BT_MESH_PROV=y
+CONFIG_NET_BUF_POOL_USAGE=y
+CONFIG_BT_MESH_PROXY=y
+CONFIG_BT_MESH_PB_GATT=y
+CONFIG_BT_MESH_GATT_PROXY=y
+CONFIG_BT_MESH_NODE_ID_TIMEOUT=60
+CONFIG_BT_MESH_PROXY_FILTER_SIZE=1
+CONFIG_BT_MESH_SUBNET_COUNT=1
+CONFIG_BT_MESH_APP_KEY_COUNT=1
+CONFIG_BT_MESH_MODEL_KEY_COUNT=1
+CONFIG_BT_MESH_MODEL_GROUP_COUNT=1
+CONFIG_BT_MESH_LABEL_COUNT=1
+CONFIG_BT_MESH_CRPL=10
+CONFIG_BT_MESH_MSG_CACHE_SIZE=10
+CONFIG_BT_MESH_ADV_BUF_COUNT=20
+CONFIG_BT_MESH_TX_SEG_MSG_COUNT=6
+CONFIG_BT_MESH_RX_SEG_MSG_COUNT=1
+CONFIG_BT_MESH_RX_SDU_MAX=384
+CONFIG_BT_MESH_RELAY=y
+CONFIG_BT_MESH_LOW_POWER=
+CONFIG_BT_MESH_FRIEND=
+CONFIG_BT_MESH_CFG_CLI=y
+CONFIG_BT_MESH_HEALTH_CLI=y
+CONFIG_BTU_TASK_STACK_SIZE=4512
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/Makefile b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/Makefile
new file mode 100644
index 0000000000..6c7b6101fa
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/Makefile
@@ -0,0 +1,10 @@
+#
+# This is a project Makefile. It is assumed the directory this Makefile resides in is a
+# project subdirectory.
+#
+
+PROJECT_NAME := ble_mesh_console_provisioner
+
+COMPONENT_ADD_INCLUDEDIRS := components/include
+
+include $(IDF_PATH)/make/project.mk
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/README.md b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/README.md
new file mode 100644
index 0000000000..3925d93cf4
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/README.md
@@ -0,0 +1,10 @@
+# ble mesh provisioner demo
+## Introduction
+This demo implements ble mesh provisioner basic features.Based on this demo, provisioner can scan and prove unprovisioned device, send set/get message. Also can define new model.
+
+Demo steps:
+1. Build the ble mesh provisioner demo with sdkconfig.default
+2. register provisioner and set oob info, load model to init ble mesh provisioner
+3. enable bearer, so that it can scan and prove unprovisioned devices
+4. config appkey and other config info use config client model
+5. send set/get message to nodes
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_adapter.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_adapter.c
new file mode 100644
index 0000000000..a7a55d605f
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_adapter.c
@@ -0,0 +1,300 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "esp_ble_mesh_networking_api.h"
+#include "ble_mesh_adapter.h"
+
+esp_ble_mesh_model_t *ble_mesh_get_model(uint16_t model_id)
+{
+ esp_ble_mesh_model_t *model = NULL;
+
+ switch (model_id) {
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_SRV:
+ model = config_server_models;
+ break;
+#if (CONFIG_BT_MESH_CFG_CLI)
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_CLI:
+ model = &gen_onoff_cli_models[1];
+ break;
+#endif
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV:
+ model = &gen_onoff_srv_models[1];
+ break;
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_CLI:
+ model = &gen_onoff_cli_models[2];
+ break;
+#endif
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI:
+ model = &test_perf_cli_models[0];
+ break;
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV:
+ model = &test_perf_srv_models[0];
+ break;
+ }
+
+ return model;
+}
+
+esp_ble_mesh_comp_t *ble_mesh_get_component(uint16_t model_id)
+{
+ esp_ble_mesh_comp_t *comp = NULL;
+
+ switch (model_id) {
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_SRV:
+ comp = &config_server_comp;
+ break;
+ case ESP_BLE_MESH_MODEL_ID_CONFIG_CLI:
+ comp = &config_client_comp;
+ break;
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV:
+ comp = &gen_onoff_srv_comp;
+ break;
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+ case ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_CLI:
+ comp = &gen_onoff_cli_comp;
+ break;
+#endif
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI:
+ comp = &test_perf_cli_comp;
+ break;
+ case ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV:
+ comp = &test_perf_srv_comp;
+ break;
+ }
+
+ return comp;
+}
+
+void ble_mesh_node_init()
+{
+ uint16_t i;
+
+ for (i = 0; i < NODE_MAX_GROUP_CONFIG; i++) {
+ ble_mesh_node_prestore_params[i].net_idx = 0xFFFF;
+ ble_mesh_node_prestore_params[i].unicast_addr = 0xFFFF;
+ }
+
+ ble_mesh_node_sema = xSemaphoreCreateMutex();
+ if (!ble_mesh_node_sema) {
+ ESP_LOGE(TAG, "%s failed to init, failed to create mesh node semaphore", __func__);
+ }
+}
+
+void ble_mesh_set_node_prestore_params(uint16_t netkey_index, uint16_t unicast_addr)
+{
+ uint16_t i;
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY);
+ for (i = 0; i < NODE_MAX_GROUP_CONFIG; i++) {
+ if (ble_mesh_node_prestore_params[i].net_idx != 0xFFFF && ble_mesh_node_prestore_params[i].unicast_addr != 0xFFFF) {
+ ble_mesh_node_prestore_params[i].net_idx = netkey_index;
+ ble_mesh_node_prestore_params[i].unicast_addr = unicast_addr;
+ }
+ }
+ xSemaphoreGive(ble_mesh_node_sema);
+}
+
+void ble_mesh_create_send_data(char *data, uint16_t byte_num, uint16_t sequence_num, uint32_t opcode)
+{
+ uint16_t i;
+
+ // first two bytes are sequence num, third is type
+ data[0] = sequence_num >> 8;
+ data[1] = sequence_num & 0xFF;
+ switch (opcode) {
+ case ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET:
+ data[2] = VENDOR_MODEL_PERF_OPERATION_TYPE_GET;
+ break;
+ case ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET:
+ data[2] = VENDOR_MODEL_PERF_OPERATION_TYPE_SET;
+ break;
+ case ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK:
+ data[2] = VENDOR_MODEL_PERF_OPERATION_TYPE_SET_UNACK;
+ break;
+ }
+
+ for (i = 3; i < byte_num; i++) {
+ data[i] = i;
+ }
+}
+
+void ble_mesh_test_performance_client_model_get()
+{
+ uint32_t i, j;
+ uint32_t sum_time = 0;
+
+ xSemaphoreTake(ble_mesh_test_perf_sema, portMAX_DELAY);
+
+ for (i = 0, j = 0; i < test_perf_statistics.test_num; i++) {
+ if (test_perf_statistics.time[i] != 0) {
+ sum_time += test_perf_statistics.time[i];
+ j += 1;
+ } else {
+ continue;
+ }
+
+ if (j == test_perf_statistics.test_num - 1) {
+ break;
+ }
+ }
+
+ ESP_LOGI(TAG, "VendorModel:Statistics,%d,%d\n",
+ test_perf_statistics.statistics, (sum_time / (j + 1)));
+
+ xSemaphoreGive(ble_mesh_test_perf_sema);
+}
+
+void ble_mesh_test_performance_client_model_get_received_percent()
+{
+ uint32_t i, j;
+ uint32_t max_time = 1400;
+ uint32_t min_time = 0;
+ uint32_t time_level_num = 0;
+ typedef struct {
+ uint16_t time_level;
+ uint16_t time_num;
+ } statistics_time_performance;
+ statistics_time_performance *statistics_time_percent;
+
+ xSemaphoreTake(ble_mesh_test_perf_sema, portMAX_DELAY);
+
+ time_level_num = ((max_time - min_time) / 50 + 1);
+ statistics_time_percent = malloc(sizeof(statistics_time_performance) * time_level_num);
+
+ for (j = 0; j < time_level_num; j++) {
+ statistics_time_percent[j].time_level = min_time + 50 * j;
+ statistics_time_percent[j].time_num = 0;
+ }
+
+ for (i = 0; i < test_perf_statistics.test_num; i++) {
+ for (j = 0; j < time_level_num; j++) {
+ if (test_perf_statistics.time[i] > max_time) {
+ j -= 1;
+ break;
+ }
+ if (test_perf_statistics.time[i] >= min_time + 50 * j
+ && test_perf_statistics.time[i] < min_time + 50 * (j + 1)) {
+ statistics_time_percent[j].time_num += 1;
+ break;
+ }
+ }
+ }
+
+ // for script match
+ ESP_LOGI(TAG, "VendorModel:Statistics");
+ for (j = 0; j < time_level_num; j++) {
+ printf(",%d:%d", statistics_time_percent[j].time_level, statistics_time_percent[j].time_num);
+ }
+ printf("\n");
+
+ free(statistics_time_percent);
+ xSemaphoreGive(ble_mesh_test_perf_sema);
+}
+
+void ble_mesh_test_performance_client_model_accumulate_statistics(uint32_t value)
+{
+ xSemaphoreTake(ble_mesh_test_perf_sema, portMAX_DELAY);
+ test_perf_statistics.statistics += value;
+ xSemaphoreGive(ble_mesh_test_perf_sema);
+}
+
+int ble_mesh_test_performance_client_model_accumulate_time(uint16_t time, uint8_t *data, uint8_t ack_ttl, uint16_t length)
+{
+ uint16_t i;
+ uint16_t sequence_num = 0;
+ uint16_t node_received_ttl = 0;
+ xSemaphoreTake(ble_mesh_test_perf_sema, portMAX_DELAY);
+
+ // received fail
+ if (length != test_perf_statistics.test_length) {
+ xSemaphoreGive(ble_mesh_test_perf_sema);
+ return 1;
+ }
+
+ if (data != NULL) {
+ sequence_num = (data[0] << 8) | data[1];
+ if (data[2] == VENDOR_MODEL_PERF_OPERATION_TYPE_SET) {
+ node_received_ttl = data[3];
+ }
+ }
+
+ for (i = 0; i < test_perf_statistics.test_num; i++) {
+ if (test_perf_statistics.package_index[i] == sequence_num) {
+ xSemaphoreGive(ble_mesh_test_perf_sema);
+ return 1;
+ }
+ }
+
+ for (i = 0; i < test_perf_statistics.test_num; i++) {
+ if (test_perf_statistics.package_index[i] == 0) {
+ test_perf_statistics.package_index[i] = sequence_num;
+ if (data[2] == VENDOR_MODEL_PERF_OPERATION_TYPE_SET) {
+ if (node_received_ttl == test_perf_statistics.ttl && ack_ttl == test_perf_statistics.ttl) {
+ test_perf_statistics.time[i] = time;
+ } else {
+ test_perf_statistics.time[i] = 0;
+ }
+ } else if (data[2] == VENDOR_MODEL_PERF_OPERATION_TYPE_SET_UNACK) {
+ test_perf_statistics.time[i] = time;
+ }
+ break;
+ }
+ }
+
+ xSemaphoreGive(ble_mesh_test_perf_sema);
+ return 0;
+}
+
+int ble_mesh_test_performance_client_model_init(uint16_t node_num, uint32_t test_num, uint8_t ttl)
+{
+ uint16_t i;
+
+ // malloc time
+ test_perf_statistics.time = malloc(test_num * sizeof(uint16_t));
+ if (test_perf_statistics.time == NULL) {
+ ESP_LOGE(TAG, " %s %d, malloc fail\n", __func__, __LINE__);
+ return 1;
+ }
+
+ test_perf_statistics.package_index = malloc(test_num * sizeof(uint16_t));
+ if (test_perf_statistics.package_index == NULL) {
+ ESP_LOGE(TAG, " %s %d, malloc fail\n", __func__, __LINE__);
+ }
+ for (i = 0; i < test_num; i++) {
+ test_perf_statistics.time[i] = 0;
+ test_perf_statistics.package_index[i] = 0;
+ }
+
+ test_perf_statistics.test_num = test_num;
+ test_perf_statistics.node_num = node_num;
+ test_perf_statistics.ttl = ttl;
+ test_perf_statistics.statistics = 0;
+ return 0;
+}
+
+void ble_mesh_test_performance_client_model_destroy()
+{
+ if (test_perf_statistics.time != NULL) {
+ free(test_perf_statistics.time);
+ }
+
+ if (test_perf_statistics.package_index != NULL) {
+ free(test_perf_statistics.package_index);
+ }
+
+ test_perf_statistics.test_num = 0;
+ test_perf_statistics.ttl = 0;
+ test_perf_statistics.node_num = 0;
+ test_perf_statistics.statistics = 0;
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_adapter.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_adapter.h
new file mode 100644
index 0000000000..87bc7399a4
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_adapter.h
@@ -0,0 +1,123 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BLE_MESH_ADAPTER_H_
+#define _BLE_MESH_ADAPTER_H_
+
+#include "freertos/FreeRTOS.h"
+#include "freertos/semphr.h"
+
+#include "ble_mesh_console_lib.h"
+#include "ble_mesh_cfg_srv_model.h"
+
+#define TAG "ble_mesh_prov_console"
+
+uint64_t start_time;
+typedef enum {
+ VENDOR_MODEL_PERF_OPERATION_TYPE_GET = 1,
+ VENDOR_MODEL_PERF_OPERATION_TYPE_SET,
+ VENDOR_MODEL_PERF_OPERATION_TYPE_SET_UNACK
+} ble_mesh_perf_operation_type;
+
+typedef struct {
+ uint8_t current;
+ uint8_t previous;
+ char *name;
+} ble_mesh_node_status;
+
+typedef struct {
+ bool need_ack;
+ uint8_t ttl;
+ uint16_t length;
+ uint16_t test_num;
+ uint16_t address;
+ uint16_t app_idx;
+ uint16_t net_idx;
+ uint32_t opcode;
+ esp_ble_mesh_model_t *model;
+ esp_ble_mesh_dev_role_t device_role;
+} ble_mesh_test_perf_throughput_data;
+
+typedef struct {
+ uint32_t statistics;
+ uint32_t test_num;
+ uint16_t test_length;
+ uint16_t node_num;
+ uint16_t *time;
+ uint16_t *package_index;
+ uint8_t ttl;
+} ble_mesh_performance_statistics_t;
+ble_mesh_performance_statistics_t test_perf_statistics;
+
+#define SEND_MESSAGE_TIMEOUT (30000/portTICK_RATE_MS)
+
+extern SemaphoreHandle_t ble_mesh_node_sema;
+extern SemaphoreHandle_t ble_mesh_test_perf_send_sema;
+extern SemaphoreHandle_t ble_mesh_test_perf_sema;
+
+#define arg_int_to_value(src_msg, dst_msg, message) do { \
+ if (src_msg->count != 0) {\
+ ESP_LOGD(TAG, " %s, %s\n", __func__, message);\
+ dst_msg = src_msg->ival[0];\
+ } \
+} while(0) \
+
+#define ble_mesh_node_get_value(index, key, value) do { \
+ uint16_t _index = 0; \
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY); \
+ for (_index = 0; _index < NODE_MAX_GROUP_CONFIG; _index) { \
+ if (node_set_prestore_params[_index].key == value) { \
+ break; \
+ } \
+ } \
+ index = _index; \
+ xSemaphoreGive(ble_mesh_node_sema); \
+} while(0) \
+
+#define ble_mesh_node_set_state(status) do { \
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY); \
+ node_status.previous = node_status.current; \
+ node_status.current = status; \
+ xSemaphoreGive(ble_mesh_node_sema); \
+}while(0) \
+
+#define ble_mesh_node_get_state(status) do { \
+ xSemaphoreTake(ble_mesh_node_sema, portMAX_DELAY); \
+ status = node_status.previous; \
+ xSemaphoreGive(ble_mesh_node_sema); \
+}while(0) \
+
+#define ble_mesh_callback_check_err_code(err_code, message) do { \
+ if (err_code == ESP_OK) { \
+ ESP_LOGI(TAG, "%s,OK\n", message); \
+ } else { \
+ ESP_LOGI(TAG, "%s,Fail,%d\n", message, err_code); \
+ } \
+}while(0) \
+
+void ble_mesh_node_init();
+void ble_mesh_set_node_prestore_params(uint16_t netkey_index, uint16_t unicast_addr);
+
+esp_ble_mesh_model_t *ble_mesh_get_model(uint16_t model_id);
+esp_ble_mesh_comp_t *ble_mesh_get_component(uint16_t model_id);
+void ble_mesh_create_send_data(char *data, uint16_t byte_num, uint16_t sequence_num, uint32_t opcode);
+
+void ble_mesh_test_performance_client_model_get();
+void ble_mesh_test_performance_client_model_get_received_percent();
+void ble_mesh_test_performance_client_model_accumulate_statistics(uint32_t value);
+int ble_mesh_test_performance_client_model_accumulate_time(uint16_t time, uint8_t *data, uint8_t ack_ttl, uint16_t length);
+int ble_mesh_test_performance_client_model_init(uint16_t node_num, uint32_t test_num, uint8_t ttl);
+void ble_mesh_test_performance_client_model_destroy();
+
+#endif //_BLE_MESH_ADAPTER_H_
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_cfg_srv_model.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_cfg_srv_model.c
new file mode 100644
index 0000000000..a0128e46f7
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_cfg_srv_model.c
@@ -0,0 +1,208 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "ble_mesh_cfg_srv_model.h"
+
+uint8_t dev_uuid[16] = {0xdd, 0xdd};
+
+#if CONFIG_BT_MESH_NODE
+esp_ble_mesh_prov_t prov = {
+ .uuid = dev_uuid,
+};
+#endif //CONFIG_BT_MESH_NODE
+
+#if CONFIG_BT_MESH_PROVISIONER
+esp_ble_mesh_prov_t prov = {
+ .prov_uuid = dev_uuid,
+ .prov_unicast_addr = 0x0001,
+ .prov_start_address = 0x0005,
+ .prov_attention = 0x00,
+ .prov_algorithm = 0x00,
+ .prov_pub_key_oob = 0x00,
+ .prov_static_oob_val = NULL,
+ .prov_static_oob_len = 0x00,
+ .flags = 0x00,
+ .iv_index = 0x00,
+};
+#endif //CONFIG_BT_MESH_PROVISIONER
+
+esp_ble_mesh_model_pub_t model_pub_config = {
+ .msg = NET_BUF_SIMPLE(2 + 1),
+ .update = NULL,
+};
+
+esp_ble_mesh_model_pub_t vendor_model_pub_config;
+
+// configure server module
+esp_ble_mesh_cfg_srv_t cfg_srv = {
+ .relay = ESP_BLE_MESH_RELAY_ENABLED,
+ .beacon = ESP_BLE_MESH_BEACON_ENABLED,
+#if defined(CONFIG_BT_MESH_FRIEND)
+ .friend_state = BT_MESH_FRIEND_ENABLED,
+#else
+ .friend_state = BT_MESH_FRIEND_NOT_SUPPORTED,
+#endif
+#if defined(CONFIG_BT_MESH_GATT_PROXY)
+ .gatt_proxy = BT_MESH_GATT_PROXY_ENABLED,
+#else
+ .gatt_proxy = BT_MESH_GATT_PROXY_NOT_SUPPORTED,
+#endif
+ .default_ttl = 7,
+
+ /* 3 transmissions with 20ms interval */
+ .net_transmit = ESP_BLE_MESH_TRANSMIT(2, 20),
+ .relay_retransmit = BT_MESH_TRANSMIT(0, 20),
+};
+
+esp_ble_mesh_model_t config_server_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+};
+
+esp_ble_mesh_elem_t config_server_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_server_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t config_server_comp = {
+ .cid = CID_ESP,
+ .elements = config_server_elements,
+ .element_count = ARRAY_SIZE(config_server_elements),
+};
+
+// config client model
+esp_ble_mesh_model_t config_client_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_MODEL_CFG_CLI(&cfg_cli),
+};
+
+esp_ble_mesh_elem_t config_client_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_client_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t config_client_comp = {
+ .cid = CID_ESP,
+ .elements = config_client_elements,
+ .element_count = ARRAY_SIZE(config_client_elements),
+};
+
+// configure special module
+esp_ble_mesh_model_op_t gen_onoff_srv_model_op_config[] = {
+ {ESP_BLE_MESH_MODEL_OP_2(0x82, 0x01), 0, 0},
+ {ESP_BLE_MESH_MODEL_OP_2(0x82, 0x02), 2, 0},
+ {ESP_BLE_MESH_MODEL_OP_2(0x82, 0x03), 2, 0},
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+esp_ble_mesh_model_t gen_onoff_srv_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_SIG_MODEL(ESP_BLE_MESH_MODEL_ID_GEN_ONOFF_SRV, gen_onoff_srv_model_op_config, &model_pub_config, NULL),
+};
+
+esp_ble_mesh_elem_t gen_onoff_srv_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, gen_onoff_srv_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t gen_onoff_srv_comp = {
+ .cid = CID_ESP,
+ .elements = gen_onoff_srv_elements,
+ .element_count = ARRAY_SIZE(gen_onoff_srv_elements),
+};
+
+// config generic onoff client
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+
+esp_ble_mesh_client_t gen_onoff_cli;
+
+esp_ble_mesh_model_t gen_onoff_cli_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_MODEL_CFG_CLI(&cfg_cli),
+ ESP_BLE_MESH_MODEL_GEN_ONOFF_CLI(&model_pub_config, &gen_onoff_cli),
+};
+
+esp_ble_mesh_elem_t gen_onoff_cli_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, gen_onoff_cli_models, ESP_BLE_MESH_MODEL_NONE),
+};
+
+esp_ble_mesh_comp_t gen_onoff_cli_comp = {
+ .cid = CID_ESP,
+ .elements = gen_onoff_cli_elements,
+ .element_count = ARRAY_SIZE(gen_onoff_cli_elements),
+};
+#endif //CONFIG_BT_MESH_GENERIC_ONOFF_CLI
+
+//CONFIG VENDOR MODEL TEST PERFORMANCE
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV 0x2000
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI 0x2001
+
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET ESP_BLE_MESH_MODEL_OP_3(0x01, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET ESP_BLE_MESH_MODEL_OP_3(0x02, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK ESP_BLE_MESH_MODEL_OP_3(0x03, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS ESP_BLE_MESH_MODEL_OP_3(0x04, CID_ESP)
+
+esp_ble_mesh_client_op_pair_t test_perf_cli_op_pair[] = {
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET, ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET, ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK, ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS},
+};
+
+esp_ble_mesh_client_t test_perf_cli = {
+ .op_pair_size = ARRAY_SIZE(test_perf_cli_op_pair),
+ .op_pair = test_perf_cli_op_pair,
+};
+
+esp_ble_mesh_model_op_t test_perf_srv_op[] = {
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET, 1, 0},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET, 1, 0},
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK, 1, 0},
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+esp_ble_mesh_model_op_t test_perf_cli_op[] = {
+ {ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS, 1, 0},
+ ESP_BLE_MESH_MODEL_OP_END,
+};
+
+esp_ble_mesh_model_t config_models[] = {
+ ESP_BLE_MESH_MODEL_CFG_SRV(&cfg_srv),
+ ESP_BLE_MESH_MODEL_CFG_CLI(&cfg_cli),
+};
+
+esp_ble_mesh_model_t test_perf_cli_models[] = {
+ ESP_BLE_MESH_VENDOR_MODEL(CID_ESP, ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI,
+ test_perf_cli_op, &vendor_model_pub_config, &test_perf_cli),
+};
+
+esp_ble_mesh_elem_t test_perf_cli_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_models, test_perf_cli_models),
+};
+
+esp_ble_mesh_comp_t test_perf_cli_comp = {
+ .cid = CID_ESP,
+ .elements = test_perf_cli_elements,
+ .element_count = ARRAY_SIZE(test_perf_cli_elements),
+};
+
+esp_ble_mesh_model_t test_perf_srv_models[] = {
+ ESP_BLE_MESH_VENDOR_MODEL(CID_ESP, ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV,
+ test_perf_srv_op, NULL, NULL),
+};
+
+esp_ble_mesh_elem_t test_perf_srv_elements[] = {
+ ESP_BLE_MESH_ELEMENT(0, config_models, test_perf_srv_models),
+};
+
+esp_ble_mesh_comp_t test_perf_srv_comp = {
+ .cid = CID_ESP,
+ .elements = test_perf_srv_elements,
+ .element_count = ARRAY_SIZE(test_perf_srv_elements),
+};
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_cfg_srv_model.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_cfg_srv_model.h
new file mode 100644
index 0000000000..72f012506f
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_cfg_srv_model.h
@@ -0,0 +1,109 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BLE_MESH_CFG_SRV_MODEL_H_
+#define _BLE_MESH_CFG_SRV_MODEL_H_
+
+#include "esp_ble_mesh_defs.h"
+#include "esp_ble_mesh_config_model_api.h"
+
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+#include "esp_ble_mesh_generic_model_api.h"
+#endif //CONFIG_BT_MESH_GENERIC_ONOFF_CLI
+
+#define NODE_MAX_GROUP_CONFIG 3
+#define CID_ESP 0x02C4
+
+extern uint8_t dev_uuid[16];
+
+typedef struct {
+ uint16_t net_idx;
+ uint16_t unicast_addr;
+} ble_mesh_node_config_params;
+ble_mesh_node_config_params ble_mesh_node_prestore_params[NODE_MAX_GROUP_CONFIG];
+
+extern esp_ble_mesh_prov_t prov;
+
+extern esp_ble_mesh_model_pub_t model_pub_config;
+
+extern esp_ble_mesh_model_pub_t vendor_model_pub_config;
+
+// configure server module
+extern esp_ble_mesh_cfg_srv_t cfg_srv;
+
+extern esp_ble_mesh_model_t config_server_models[];
+
+extern esp_ble_mesh_elem_t config_server_elements[];
+
+extern esp_ble_mesh_comp_t config_server_comp;
+
+// config client model
+esp_ble_mesh_client_t cfg_cli;
+extern esp_ble_mesh_model_t config_client_models[];
+
+extern esp_ble_mesh_elem_t config_client_elements[];
+
+extern esp_ble_mesh_comp_t config_client_comp;
+
+// configure special module
+extern esp_ble_mesh_model_op_t gen_onoff_srv_model_op_config[];
+
+extern esp_ble_mesh_model_t gen_onoff_srv_models[];
+
+extern esp_ble_mesh_elem_t gen_onoff_srv_elements[];
+
+extern esp_ble_mesh_comp_t gen_onoff_srv_comp;
+
+// config generic onoff client
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+
+extern esp_ble_mesh_client_t gen_onoff_cli;
+
+extern esp_ble_mesh_model_t gen_onoff_cli_models[];
+
+extern esp_ble_mesh_elem_t gen_onoff_cli_elements[];
+
+extern esp_ble_mesh_comp_t gen_onoff_cli_comp;
+#endif //CONFIG_BT_MESH_GENERIC_ONOFF_CLI
+
+//CONFIG VENDOR MODEL TEST PERFORMANCE
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_SRV 0x2000
+#define ESP_BLE_MESH_VND_MODEL_ID_TEST_PERF_CLI 0x2001
+
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_GET ESP_BLE_MESH_MODEL_OP_3(0x01, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET ESP_BLE_MESH_MODEL_OP_3(0x02, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_SET_UNACK ESP_BLE_MESH_MODEL_OP_3(0x03, CID_ESP)
+#define ESP_BLE_MESH_VND_MODEL_OP_TEST_PERF_STATUS ESP_BLE_MESH_MODEL_OP_3(0x04, CID_ESP)
+
+extern esp_ble_mesh_client_t test_perf_cli;
+
+extern esp_ble_mesh_model_op_t test_perf_srv_op[];
+
+extern esp_ble_mesh_model_op_t test_perf_cli_op[];
+
+extern esp_ble_mesh_model_t config_models[];
+
+extern esp_ble_mesh_model_t test_perf_cli_models[];
+
+extern esp_ble_mesh_elem_t test_perf_cli_elements[];
+
+extern esp_ble_mesh_comp_t test_perf_cli_comp;
+
+extern esp_ble_mesh_model_t test_perf_srv_models[];
+
+extern esp_ble_mesh_elem_t test_perf_srv_elements[];
+
+extern esp_ble_mesh_comp_t test_perf_srv_comp;
+
+#endif //_BLE_MESH_CFG_SRV_MODEL_H_
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_decl.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_decl.h
new file mode 100644
index 0000000000..f9a2ce408d
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_decl.h
@@ -0,0 +1,38 @@
+/* Console example — declarations of command registration functions.
+
+ This example code is in the Public Domain (or CC0 licensed, at your option).
+
+ Unless required by applicable law or agreed to in writing, this
+ software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
+ CONDITIONS OF ANY KIND, either express or implied.
+*/
+#pragma once
+
+#include "esp_ble_mesh_defs.h"
+
+// Register system functions
+void register_system();
+
+// Register bluetooth
+void register_bluetooth();
+
+// Register mesh node cmd
+void ble_mesh_register_mesh_node();
+
+// Register Test Perf client cmd
+void ble_mesh_register_mesh_test_performance_client();
+
+#if (CONFIG_BT_MESH_CFG_CLI)
+// Register mesh config client operation cmd
+void ble_mesh_register_configuration_client_model();
+#endif
+
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+// Register mesh client operation cmd
+void ble_mesh_register_gen_onoff_client();
+#endif
+
+#if CONFIG_BT_MESH_PROVISIONER
+// Regitster mesh provisioner cmd
+void ble_mesh_register_mesh_provisioner();
+#endif
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_lib.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_lib.c
new file mode 100644
index 0000000000..d15ee3adc1
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_lib.c
@@ -0,0 +1,124 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "ble_mesh_console_lib.h"
+
+static int hex2num(char c);
+static int hex2byte(const char *hex);
+
+static int hex2num(char c)
+{
+ if (c >= '0' && c <= '9') {
+ return c - '0';
+ }
+ if (c >= 'a' && c <= 'f') {
+ return c - 'a' + 10;
+ }
+ if (c >= 'A' && c <= 'F') {
+ return c - 'A' + 10;
+ }
+ return -1;
+}
+
+static int hex2byte(const char *hex)
+{
+ int a, b;
+ a = hex2num(*hex++);
+ if (a < 0) {
+ return -1;
+ }
+ b = hex2num(*hex++);
+ if (b < 0) {
+ return -1;
+ }
+ return (a << 4) | b;
+}
+
+int hexstr_2_bin(const char *hex, uint8_t *buf, uint32_t len)
+{
+ uint32_t i;
+ int a;
+ const char *ipos = hex;
+ uint8_t *opos = buf;
+
+ for (i = 0; i < len; i++) {
+ a = hex2byte(ipos);
+ if (a < 0) {
+ return -1;
+ }
+ *opos ++ = a;
+ ipos += 2;
+ }
+ return 0;
+}
+
+int get_value_string(char *value_in, char *buf)
+{
+ int result = -1;
+
+ uint16_t length = strlen(value_in);
+
+ if (length > 2) {
+ if (value_in[0] == '0' && value_in[1] == 'x') {
+ buf[(length - 2) / 2] = 0;
+ result = hexstr_2_bin(&value_in[2], (uint8_t *)buf, (length - 2) / 2);
+ length = (length - 2) / 2;
+ } else {
+ strcpy(buf, value_in);
+ result = 0;
+ }
+ } else {
+ strcpy(buf, value_in);
+ result = 0;
+ }
+ return result;
+}
+
+bool str_2_mac(uint8_t *str, uint8_t *dest)
+{
+ uint8_t loop = 0;
+ uint8_t tmp = 0;
+ uint8_t *src_p = str;
+
+ if (strlen((char *)src_p) != 17) { // must be like 12:34:56:78:90:AB
+ return false;
+ }
+
+ for (loop = 0; loop < 17 ; loop++) {
+ if (loop % 3 == 2) {
+ if (src_p[loop] != ':') {
+ return false;
+ }
+
+ continue;
+ }
+
+ if ((src_p[loop] >= '0') && (src_p[loop] <= '9')) {
+ tmp = tmp * 16 + src_p[loop] - '0';
+ } else if ((src_p[loop] >= 'A') && (src_p[loop] <= 'F')) {
+ tmp = tmp * 16 + src_p[loop] - 'A' + 10;
+ } else if ((src_p[loop] >= 'a') && (src_p[loop] <= 'f')) {
+ tmp = tmp * 16 + src_p[loop] - 'a' + 10;
+ } else {
+ return false;
+ }
+
+ if (loop % 3 == 1) {
+ *dest++ = tmp;
+ tmp = 0;
+ }
+ }
+
+ return true;
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_lib.h b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_lib.h
new file mode 100644
index 0000000000..11dd05cb37
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_lib.h
@@ -0,0 +1,29 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef _BLE_MESH_CONSOLE_LIB_H_
+#define _BLE_MESH_CONSOLE_LIB_H_
+
+#include
+#include
+
+#include "esp_system.h"
+#include "esp_console.h"
+#include "argtable3/argtable3.h"
+
+bool str_2_mac(uint8_t *str, uint8_t *dest);
+int hexstr_2_bin(const char *hex, uint8_t *buf, uint32_t len);
+int get_value_string(char *value_in, char *buf);
+
+#endif //_BLE_MESH_CONSOLE_LIB_H_
\ No newline at end of file
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_main.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_main.c
new file mode 100644
index 0000000000..9298c95e24
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_main.c
@@ -0,0 +1,228 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include
+#include
+
+#include "esp_system.h"
+#include "esp_log.h"
+#include "esp_console.h"
+#include "esp_vfs_dev.h"
+#include "driver/uart.h"
+#include "linenoise/linenoise.h"
+#include "argtable3/argtable3.h"
+
+#include "esp_vfs_fat.h"
+#include "nvs.h"
+#include "nvs_flash.h"
+
+#include "esp_bt.h"
+#include "esp_bt_main.h"
+
+#include "ble_mesh_console_decl.h"
+
+#define TAG "ble_mesh_test"
+
+#if CONFIG_STORE_HISTORY
+
+#define MOUNT_PATH "/data"
+#define HISTORY_PATH MOUNT_PATH "/history.txt"
+
+static void initialize_filesystem()
+{
+ static wl_handle_t wl_handle;
+ const esp_vfs_fat_mount_config_t mount_config = {
+ .max_files = 4,
+ .format_if_mount_failed = true
+ };
+ esp_err_t err = esp_vfs_fat_spiflash_mount(MOUNT_PATH, "storage", &mount_config, &wl_handle);
+ if (err != ESP_OK) {
+ ESP_LOGE(TAG, "Failed to mount FATFS (0x%x)", err);
+ return;
+ }
+}
+#endif // CONFIG_STORE_HISTORY
+
+static void initialize_console()
+{
+ /* Disable buffering on stdin and stdout */
+ setvbuf(stdin, NULL, _IONBF, 0);
+ setvbuf(stdout, NULL, _IONBF, 0);
+
+ /* Minicom, screen, idf_monitor send CR when ENTER key is pressed */
+ esp_vfs_dev_uart_set_rx_line_endings(ESP_LINE_ENDINGS_CR);
+ /* Move the caret to the beginning of the next line on '\n' */
+ esp_vfs_dev_uart_set_tx_line_endings(ESP_LINE_ENDINGS_CRLF);
+
+ /* Install UART driver for interrupt-driven reads and writes */
+ ESP_ERROR_CHECK( uart_driver_install(CONFIG_CONSOLE_UART_NUM,
+ 256, 0, 0, NULL, 0) );
+
+ /* Tell VFS to use UART driver */
+ esp_vfs_dev_uart_use_driver(CONFIG_CONSOLE_UART_NUM);
+
+ /* Initialize the console */
+ esp_console_config_t console_config = {
+ .max_cmdline_args = 20,
+ .max_cmdline_length = 256,
+#if CONFIG_LOG_COLORS
+ .hint_color = atoi(LOG_COLOR_CYAN)
+#endif
+ };
+ ESP_ERROR_CHECK( esp_console_init(&console_config) );
+
+ /* Configure linenoise line completion library */
+ /* Enable multiline editing. If not set, long commands will scroll within
+ * a single line.
+ */
+ linenoiseSetMultiLine(1);
+
+ /* Tell linenoise where to get command completions and hints */
+ linenoiseSetCompletionCallback(&esp_console_get_completion);
+ linenoiseSetHintsCallback((linenoiseHintsCallback *) &esp_console_get_hint);
+
+ /* Set command history size */
+ linenoiseHistorySetMaxLen(100);
+
+#if CONFIG_STORE_HISTORY
+ /* Load command history from filesystem */
+ linenoiseHistoryLoad(HISTORY_PATH);
+#endif
+}
+
+
+esp_err_t bluetooth_init(void)
+{
+ esp_err_t ret;
+
+ esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT();
+ ret = esp_bt_controller_init(&bt_cfg);
+ if (ret) {
+ ESP_LOGE(TAG, "%s failed to initialize controller\n", __func__);
+ return ret;
+ }
+
+ ret = esp_bt_controller_enable(ESP_BT_MODE_BLE);
+ if (ret) {
+ ESP_LOGE(TAG, "%s failed to enable controller\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_init();
+ if (ret) {
+ ESP_LOGE(TAG, "%s failed to initialize bluetooth\n", __func__);
+ return ret;
+ }
+ ret = esp_bluedroid_enable();
+ if (ret) {
+ ESP_LOGE(TAG, "%s failed to enable bluetooth\n", __func__);
+ return ret;
+ }
+
+ return ret;
+}
+
+void app_main(void)
+{
+ esp_err_t res;
+
+ nvs_flash_init();
+
+ // init and enable bluetooth
+ res = bluetooth_init();
+ if (res) {
+ printf("esp32_bluetooth_init failed (ret %d)", res);
+ }
+
+#if CONFIG_STORE_HISTORY
+ initialize_filesystem();
+#endif
+
+ initialize_console();
+
+ /* Register commands */
+ esp_console_register_help_command();
+ register_system();
+ register_bluetooth();
+ ble_mesh_register_mesh_node();
+ ble_mesh_register_mesh_test_performance_client();
+#if (CONFIG_BT_MESH_GENERIC_ONOFF_CLI)
+ ble_mesh_register_gen_onoff_client();
+#endif
+#if (CONFIG_BT_MESH_PROVISIONER)
+ ble_mesh_register_mesh_provisioner();
+#endif
+#if (CONFIG_BT_MESH_CFG_CLI)
+ ble_mesh_register_configuration_client_model();
+#endif
+
+
+ /* Prompt to be printed before each line.
+ * This can be customized, made dynamic, etc.
+ */
+ const char *prompt = LOG_COLOR_I "esp32> " LOG_RESET_COLOR;
+
+ printf("\n"
+ "This is an example of an ESP-IDF console component.\n"
+ "Type 'help' to get the list of commands.\n"
+ "Use UP/DOWN arrows to navigate through the command history.\n"
+ "Press TAB when typing a command name to auto-complete.\n");
+
+ /* Figure out if the terminal supports escape sequences */
+ int probe_status = linenoiseProbe();
+ if (probe_status) { /* zero indicates OK */
+ printf("\n"
+ "Your terminal application does not support escape sequences.\n"
+ "Line editing and history features are disabled.\n"
+ "On Windows, try using Putty instead.\n");
+ linenoiseSetDumbMode(1);
+#if CONFIG_LOG_COLORS
+ /* Since the terminal doesn't support escape sequences,
+ * don't use color codes in the prompt.
+ */
+ prompt = "esp32> ";
+#endif //CONFIG_LOG_COLORS
+ }
+
+ /* Main loop */
+ while (true) {
+ /* Get a line using linenoise.
+ * The line is returned when ENTER is pressed.
+ */
+ char *line = linenoise(prompt);
+ if (line == NULL) { /* Ignore empty lines */
+ continue;
+ }
+ /* Add the command to the history */
+ linenoiseHistoryAdd(line);
+#if CONFIG_STORE_HISTORY
+ /* Save command history to filesystem */
+ linenoiseHistorySave(HISTORY_PATH);
+#endif
+
+ /* Try to run the command */
+ int ret;
+ esp_err_t err = esp_console_run(line, &ret);
+ if (err == ESP_ERR_NOT_FOUND) {
+ printf("Unrecognized command\n");
+ } else if (err == ESP_ERR_INVALID_ARG) {
+ // command was empty
+ } else if (err == ESP_OK && ret != ESP_OK) {
+ printf("\nCommand returned non-zero error code: 0x%x\n", ret);
+ } else if (err != ESP_OK) {
+ printf("Internal error: 0x%x\n", err);
+ }
+ /* linenoise allocates line buffer on the heap, so need to free it */
+ linenoiseFree(line);
+ }
+}
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_system.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_system.c
new file mode 100644
index 0000000000..693c865faf
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_console_system.c
@@ -0,0 +1,175 @@
+/* Console example — various system commands
+
+ This example code is in the Public Domain (or CC0 licensed, at your option).
+
+ Unless required by applicable law or agreed to in writing, this
+ software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
+ CONDITIONS OF ANY KIND, either express or implied.
+*/
+
+#include
+#include
+#include
+
+#include "esp_log.h"
+#include "esp_console.h"
+#include "esp_system.h"
+#include "esp_sleep.h"
+#include "driver/rtc_io.h"
+#include "argtable3/argtable3.h"
+
+#include "ble_mesh_console_decl.h"
+
+static void register_free();
+static void register_restart();
+static void register_make();
+
+void register_system()
+{
+ register_free();
+ register_restart();
+ register_make();
+}
+
+/** 'restart' command restarts the program */
+
+static int restart(int argc, char **argv)
+{
+ printf("%s, %s", __func__, "Restarting");
+ esp_restart();
+}
+
+static void register_restart()
+{
+ const esp_console_cmd_t cmd = {
+ .command = "restart",
+ .help = "Restart the program",
+ .hint = NULL,
+ .func = &restart,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) );
+}
+
+/** 'free' command prints available heap memory */
+
+static int free_mem(int argc, char **argv)
+{
+ printf("%d\n", esp_get_free_heap_size());
+ return 0;
+}
+
+static void register_free()
+{
+ const esp_console_cmd_t cmd = {
+ .command = "free",
+ .help = "Get the total size of heap memory available",
+ .hint = NULL,
+ .func = &free_mem,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) );
+}
+
+static int make(int argc, char **argv)
+{
+ int count = REG_READ(RTC_CNTL_STORE0_REG);
+ if (++count >= 3) {
+ printf("This is not the console you are looking for.\n");
+ return 0;
+ }
+ REG_WRITE(RTC_CNTL_STORE0_REG, count);
+
+ const char *make_output =
+ R"(LD build/console.elf
+esptool.py v2.1-beta1
+)";
+
+ const char* flash_output[] = {
+R"(Flashing binaries to serial port )" CONFIG_ESPTOOLPY_PORT R"( (app at offset 0x10000)...
+esptool.py v2.1-beta1
+Connecting....
+)",
+R"(Chip is ESP32D0WDQ6 (revision 0)
+Uploading stub...
+Running stub...
+Stub running...
+Changing baud rate to 921600
+Changed.
+Configuring flash size...
+Auto-detected Flash size: 4MB
+Flash params set to 0x0220
+Compressed 15712 bytes to 9345...
+)",
+R"(Wrote 15712 bytes (9345 compressed) at 0x00001000 in 0.1 seconds (effective 1126.9 kbit/s)...
+Hash of data verified.
+Compressed 333776 bytes to 197830...
+)",
+R"(Wrote 333776 bytes (197830 compressed) at 0x00010000 in 3.3 seconds (effective 810.3 kbit/s)...
+Hash of data verified.
+Compressed 3072 bytes to 82...
+)",
+R"(Wrote 3072 bytes (82 compressed) at 0x00008000 in 0.0 seconds (effective 1588.4 kbit/s)...
+Hash of data verified.
+Leaving...
+Hard resetting...
+)"
+ };
+
+ const char* monitor_output =
+R"(MONITOR
+)" LOG_COLOR_W R"(--- idf_monitor on )" CONFIG_ESPTOOLPY_PORT R"( 115200 ---
+--- Quit: Ctrl+] | Menu: Ctrl+T | Help: Ctrl+T followed by Ctrl+H --
+)" LOG_RESET_COLOR;
+
+ bool need_make = false;
+ bool need_flash = false;
+ bool need_monitor = false;
+ for (int i = 1; i < argc; ++i) {
+ if (strcmp(argv[i], "all") == 0) {
+ need_make = true;
+ } else if (strcmp(argv[i], "flash") == 0) {
+ need_make = true;
+ need_flash = true;
+ } else if (strcmp(argv[i], "monitor") == 0) {
+ need_monitor = true;
+ } else if (argv[i][0] == '-') {
+ /* probably -j option */
+ } else if (isdigit((int) argv[i][0])) {
+ /* might be an argument to -j */
+ } else {
+ printf("make: *** No rule to make target `%s'. Stop.\n", argv[i]);
+ /* Technically this is an error, but let's not spoil the output */
+ return 0;
+ }
+ }
+ if (argc == 1) {
+ need_make = true;
+ }
+ if (need_make) {
+ printf("%s", make_output);
+ }
+ if (need_flash) {
+ size_t n_items = sizeof(flash_output) / sizeof(flash_output[0]);
+ for (int i = 0; i < n_items; ++i) {
+ printf("%s", flash_output[i]);
+ vTaskDelay(200/portTICK_PERIOD_MS);
+ }
+ }
+ if (need_monitor) {
+ printf("%s", monitor_output);
+ esp_restart();
+ }
+ return 0;
+}
+
+static void register_make()
+{
+ const esp_console_cmd_t cmd = {
+ .command = "make",
+ .help = NULL, /* Do not include in 'help' output */
+ .hint = "all | flash | monitor",
+ .func = &make,
+ };
+ ESP_ERROR_CHECK( esp_console_cmd_register(&cmd) );
+}
+
+
diff --git a/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_reg_cfg_client_cmd.c b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_reg_cfg_client_cmd.c
new file mode 100644
index 0000000000..ba2c630d94
--- /dev/null
+++ b/examples/bluetooth/ble_mesh/ble_mesh_console/ble_mesh_provisioner/main/ble_mesh_reg_cfg_client_cmd.c
@@ -0,0 +1,390 @@
+// Copyright 2017-2018 Espressif Systems (Shanghai) PTE LTD
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "esp_ble_mesh_networking_api.h"
+#include "ble_mesh_adapter.h"
+
+#if (CONFIG_BT_MESH_CFG_CLI)
+typedef struct {
+ struct arg_str *action_type;
+ struct arg_str *set_state;
+ struct arg_int *opcode;
+ struct arg_int *unicast_address;
+ struct arg_int *appkey_index;
+ struct arg_int *mod_id;
+ struct arg_int *addr;
+ struct arg_int *cid;
+ struct arg_int *value;
+ struct arg_int *relay_statue;
+ struct arg_int *relay_transmit;
+ struct arg_int *net_idx;
+ struct arg_end *end;
+} ble_mesh_client_get_set_state_t;
+ble_mesh_client_get_set_state_t configuration_client_model_operation;
+
+void ble_mesh_register_configuration_client_model_command();
+void ble_mesh_configuration_client_model_cb(esp_ble_mesh_cfg_client_cb_event_t event,
+ esp_ble_mesh_cfg_client_cb_param_t *param);
+
+void ble_mesh_register_configuration_client_model()
+{
+ ble_mesh_register_configuration_client_model_command();
+}
+
+void ble_mesh_configuration_client_model_cb(esp_ble_mesh_cfg_client_cb_event_t event,
+ esp_ble_mesh_cfg_client_cb_param_t *param)
+{
+ uint32_t opcode;
+ ESP_LOGD(TAG, "enter %s, event = %x\n, error_code = %x\n", __func__, event, param->error_code);
+
+ if (!param->error_code) {
+ opcode = param->params->opcode;
+ switch (event) {
+ case ESP_BLE_MESH_CFG_CLIENT_GET_STATE_EVT:
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_BEACON_GET:
+ ESP_LOGI(TAG, "CfgClient:beacon,0x%x", param->status_cb.beacon_status.beacon);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_COMPOSITION_DATA_GET:
+ ESP_LOGI(TAG, "CfgClient:page,0x%x,len,0x%x", param->status_cb.comp_data_status.page, param->status_cb.comp_data_status.composition_data->len);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_GET:
+ ESP_LOGI(TAG, "CfgClient:ttl,0x%x", param->status_cb.default_ttl_status.default_ttl);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GATT_PROXY_GET:
+ ESP_LOGI(TAG, "CfgClient:proxy,0x%x", param->status_cb.gatt_proxy_status.gatt_proxy);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_RELAY_GET:
+ ESP_LOGI(TAG, "CfgClient:relay,0x%x,retransmit,0x%x", param->status_cb.relay_status.relay, param->status_cb.relay_status.retransmit);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_PUB_GET:
+ if (param->status_cb.model_pub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CfgClient:PublishGet,OK,0x%x", param->status_cb.model_pub_status.publish_addr);
+ } else {
+ ESP_LOGI(TAG, "CfgClient:PublishGet,Fail");
+ }
+
+ break;
+ case ESP_BLE_MESH_MODEL_OP_FRIEND_GET:
+ ESP_LOGI(TAG, "CfgClient:friend,0x%x", param->status_cb.friend_status.friend_state);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_GET:
+ if (param->status_cb.heartbeat_pub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatPubGet,OK,destination:0x%x,countlog:0x%x,periodlog:0x%x,ttl:0x%x,features:0x%x,net_idx:0x%x",
+ param->status_cb.heartbeat_pub_status.dst, param->status_cb.heartbeat_pub_status.count, param->status_cb.heartbeat_pub_status.period,
+ param->status_cb.heartbeat_pub_status.ttl, param->status_cb.heartbeat_pub_status.features, param->status_cb.heartbeat_pub_status.net_idx);
+ } else {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatGet,Fail,%d", param->status_cb.heartbeat_pub_status.status);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_GET:
+ if (param->status_cb.heartbeat_sub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatSubGet,OK,source:0x%x,destination:0x%x, periodlog:0x%x,countlog:0x%x,minhops:0x%x,maxhops:0x%x",
+ param->status_cb.heartbeat_sub_status.src, param->status_cb.heartbeat_sub_status.dst, param->status_cb.heartbeat_sub_status.period,
+ param->status_cb.heartbeat_sub_status.count, param->status_cb.heartbeat_sub_status.min_hops, param->status_cb.heartbeat_sub_status.max_hops);
+ } else {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatSubGet,Fail,%d", param->status_cb.heartbeat_sub_status.status);
+ }
+ break;
+ default:
+ ESP_LOGI(TAG, "Not supported config client get message opcode");
+ break;
+ }
+ break;
+ case ESP_BLE_MESH_CFG_CLIENT_SET_STATE_EVT:
+ switch (opcode) {
+ case ESP_BLE_MESH_MODEL_OP_BEACON_SET:
+ ESP_LOGI(TAG, "CfgClient:beacon,0x%x", param->status_cb.beacon_status.beacon);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_DEFAULT_TTL_SET:
+ ESP_LOGI(TAG, "CfgClient:ttl,0x%x", param->status_cb.default_ttl_status.default_ttl);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_GATT_PROXY_SET:
+ ESP_LOGI(TAG, "CfgClient:proxy,0x%x", param->status_cb.gatt_proxy_status.gatt_proxy);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_RELAY_SET:
+ ESP_LOGI(TAG, "CfgClient:relay,0x%x, retransmit: 0x%x", param->status_cb.relay_status.relay, param->status_cb.relay_status.retransmit);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_PUB_SET:
+ if (param->status_cb.model_pub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CfgClient:PublishSet,OK,0x%x", param->status_cb.model_pub_status.publish_addr);
+ } else {
+ ESP_LOGI(TAG, "CfgClient:PublishSet,Fail");
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_ADD:
+ if (param->status_cb.model_sub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CnfClient:SubAdd,OK,%x,%x", param->status_cb.model_sub_status.element_addr, param->status_cb.model_sub_status.sub_addr);
+ } else {
+ ESP_LOGI(TAG, "CnfClient:SubAdd,Fail,%x", param->status_cb.model_sub_status.status);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_DELETE:
+ if (param->status_cb.model_sub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CnfClient:SubDel,OK,%x,%x", param->status_cb.model_sub_status.element_addr, param->status_cb.model_sub_status.sub_addr);
+ } else {
+ ESP_LOGI(TAG, "CnfClient:SubDel,Fail,%x", param->status_cb.model_sub_status.status);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_OVERWRITE:
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_ADD:
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_DELETE:
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_SUB_VIRTUAL_ADDR_OVERWRITE:
+ break;
+ case ESP_BLE_MESH_MODEL_OP_NET_KEY_ADD:
+ if (param->status_cb.netkey_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CfgClient:NetKeyAdd,OK");
+ } else {
+ ESP_LOGI(TAG, "CfgClient:NetKeyAdd,Fail,%d", param->status_cb.netkey_status.status);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_APP_KEY_ADD:
+ if (param->status_cb.appkey_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CnfClient:AddAppkey,OK,%x,%x,%x", param->status_cb.appkey_status.net_idx, param->status_cb.appkey_status.app_idx, param->params->ctx.addr);
+ } else {
+ ESP_LOGI(TAG, "CnfClient:AddAppkey,Fail,%x", param->status_cb.appkey_status.status);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_MODEL_APP_BIND:
+ if (param->status_cb.model_app_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CnfClient:AppkeyBind,OK,%x,%x,%x", param->status_cb.model_app_status.app_idx, param->status_cb.model_app_status.model_id, param->params->ctx.addr);
+ } else {
+ ESP_LOGI(TAG, "CnfClient:AppkeyBind,Fail,%x", param->status_cb.model_app_status.status);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_FRIEND_SET:
+ ESP_LOGI(TAG, "CfgClient:friend: 0x%x", param->status_cb.friend_status.friend_state);
+ break;
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_PUB_SET:
+ if (param->status_cb.heartbeat_pub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatPubSet,OK,destination:0x%x,countlog:0x%x, periodlog:0x%x,ttl:0x%x,features:0x%x,net_idx: 0x%x",
+ param->status_cb.heartbeat_pub_status.dst, param->status_cb.heartbeat_pub_status.count, param->status_cb.heartbeat_pub_status.period,
+ param->status_cb.heartbeat_pub_status.ttl, param->status_cb.heartbeat_pub_status.features, param->status_cb.heartbeat_pub_status.net_idx);
+ } else {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatSet,Fail,%d", param->status_cb.heartbeat_pub_status.status);
+ }
+ break;
+ case ESP_BLE_MESH_MODEL_OP_HEARTBEAT_SUB_SET:
+ if (param->status_cb.heartbeat_sub_status.status == ESP_OK) {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatSubSet,OK,source:0x%x,destination:0x%x, periodlog:0x%x,countlog:0x%x,minhops:0x%x,maxhops:0x%x",
+ param->status_cb.heartbeat_sub_status.src, param->status_cb.heartbeat_sub_status.dst, param->status_cb.heartbeat_sub_status.period,
+ param->status_cb.heartbeat_sub_status.count, param->status_cb.heartbeat_sub_status.min_hops, param->status_cb.heartbeat_sub_status.max_hops);
+ } else {
+ ESP_LOGI(TAG, "CfgClient:HeartBeatSubSet,Fail,%d", param->status_cb.heartbeat_sub_status.status);
+ }
+ break;
+ default:
+ ESP_LOGI(TAG, "Not supported config client set message opcode");
+ break;
+ }
+ break;
+ case ESP_BLE_MESH_CFG_CLIENT_PUBLISH_EVT:
+ ESP_LOGI(TAG, "CnfClient:Publish,OK");
+ break;
+ case ESP_BLE_MESH_CFG_CLIENT_EVT_MAX:
+ ESP_LOGI(TAG, "CnfClient:MaxEvt");
+ break;
+ case ESP_BLE_MESH_CFG_CLIENT_TIMEOUT_EVT:
+ ESP_LOGI(TAG, "CfgClient:TimeOut");
+ break;
+ default:
+ ESP_LOGI(TAG, "CfgClient:InvalidEvent");
+ break;
+ }
+ } else {
+ ESP_LOGI(TAG, "CnfClient:Fail,%d", param->error_code);
+ }
+ ESP_LOGD(TAG, "exit %s \n", __func__);
+}
+
+int ble_mesh_configuration_client_model_operation(int argc, char **argv)
+{
+ int err = ESP_OK;
+ const uint8_t *app_key = NULL;
+ esp_ble_mesh_cfg_default_ttl_set_t ttl_set;
+ esp_ble_mesh_cfg_gatt_proxy_set_t proxy_set;
+ esp_ble_mesh_cfg_app_key_add_t app_key_add;
+ esp_ble_mesh_cfg_model_pub_set_t mod_pub_set = {
+ .company_id = 0xFFFF,
+ .cred_flag = false,
+ .publish_period = 0,
+ .publish_retransmit = 0,
+ };
+ esp_ble_mesh_cfg_model_sub_add_t mod_sub_add = {
+ .company_id = 0xFFFF,
+ };
+ esp_ble_mesh_cfg_model_sub_delete_t mod_sub_del = {
+ .company_id = 0xFFFF,
+ };
+ esp_ble_mesh_cfg_relay_set_t relay_set;
+ esp_ble_mesh_client_common_param_t client_common = {
+ .msg_role = ROLE_PROVISIONER,
+ .msg_timeout = 0,
+ .ctx.send_ttl = 7,
+ };
+ esp_ble_mesh_cfg_client_get_state_t get_state = {
+ .comp_data_get.page = 0,
+ .model_pub_get.company_id = 0xFFFF,
+ };
+ esp_ble_mesh_cfg_model_app_bind_t mod_app_bind = {
+ .company_id = 0xFFFF,
+ };
+
+ client_common.model = ble_mesh_get_model(ESP_BLE_MESH_MODEL_ID_CONFIG_CLI);
+
+ ESP_LOGD(TAG, "enter %s \n", __func__);
+
+ int nerrors = arg_parse(argc, argv, (void **) &configuration_client_model_operation);
+ if (nerrors != 0) {
+ arg_print_errors(stderr, configuration_client_model_operation.end, argv[0]);
+ return 1;
+ }
+
+ if (configuration_client_model_operation.opcode->count != 0) {
+ client_common.opcode = configuration_client_model_operation.opcode->ival[0];
+ }
+
+ if (configuration_client_model_operation.net_idx->count != 0) {
+ client_common.ctx.net_idx = configuration_client_model_operation.net_idx->ival[0];
+ app_key_add.net_idx = configuration_client_model_operation.net_idx->ival[0];
+ }
+
+ if (configuration_client_model_operation.unicast_address->count != 0) {
+ client_common.ctx.addr = configuration_client_model_operation.unicast_address->ival[0];
+ get_state.model_pub_get.element_addr = configuration_client_model_operation.unicast_address->ival[0];
+ mod_app_bind.element_addr = configuration_client_model_operation.unicast_address->ival[0];
+ mod_sub_add.element_addr = configuration_client_model_operation.unicast_address->ival[0];
+ mod_sub_del.element_addr = configuration_client_model_operation.unicast_address->ival[0];
+ mod_pub_set.element_addr = configuration_client_model_operation.unicast_address->ival[0];
+ }
+
+ if (configuration_client_model_operation.appkey_index->count != 0) {
+ client_common.ctx.app_idx = configuration_client_model_operation.appkey_index->ival[0];
+ mod_app_bind.model_app_idx = configuration_client_model_operation.appkey_index->ival[0];
+ app_key_add.app_idx = configuration_client_model_operation.appkey_index->ival[0];
+ mod_pub_set.publish_app_idx = configuration_client_model_operation.appkey_index->ival[0];
+ }
+
+ if (configuration_client_model_operation.value->count != 0) {
+ ttl_set.ttl = configuration_client_model_operation.value->ival[0];
+ proxy_set.gatt_proxy = configuration_client_model_operation.value->ival[0];
+ mod_pub_set.publish_ttl = configuration_client_model_operation.value->ival[0];
+ }
+
+ if (configuration_client_model_operation.addr->count != 0) {
+ mod_sub_del.sub_addr = configuration_client_model_operation.addr->ival[0];
+ mod_sub_add.sub_addr = configuration_client_model_operation.addr->ival[0];
+ mod_pub_set.publish_addr = configuration_client_model_operation.addr->ival[0];
+ }
+
+ if (configuration_client_model_operation.mod_id->count != 0) {
+ mod_app_bind.model_id = configuration_client_model_operation.mod_id->ival[0];
+ mod_sub_add.model_id = configuration_client_model_operation.mod_id->ival[0];
+ mod_sub_del.model_id = configuration_client_model_operation.mod_id->ival[0];
+ get_state.model_pub_get.model_id = configuration_client_model_operation.mod_id->ival[0];;
+ mod_pub_set.model_id = configuration_client_model_operation.mod_id->ival[0];
+ }
+
+ if (configuration_client_model_operation.relay_statue->count != 0) {
+ relay_set.relay = configuration_client_model_operation.relay_statue->ival[0];
+ mod_pub_set.publish_period = configuration_client_model_operation.relay_statue->ival[0];
+ }
+
+ if (configuration_client_model_operation.relay_transmit->count != 0) {
+ relay_set.relay_retransmit = configuration_client_model_operation.relay_transmit->ival[0];
+ mod_pub_set.publish_retransmit = configuration_client_model_operation.relay_transmit->ival[0];
+ }
+
+ if (configuration_client_model_operation.cid->count != 0) {
+ mod_app_bind.company_id = configuration_client_model_operation.cid->ival[0];
+ mod_sub_del.company_id = configuration_client_model_operation.cid->ival[0];
+ mod_sub_add.company_id = configuration_client_model_operation.cid->ival[0];
+ mod_pub_set.company_id = configuration_client_model_operation.cid->ival[0];
+ }
+
+ if (configuration_client_model_operation.action_type->count != 0) {
+ if (strcmp(configuration_client_model_operation.action_type->sval[0], "get") == 0) {
+ err = esp_ble_mesh_config_client_get_state(&client_common, &get_state);
+ } else if (strcmp(configuration_client_model_operation.action_type->sval[0], "set") == 0) {
+ if (configuration_client_model_operation.set_state->count != 0) {
+ if (strcmp(configuration_client_model_operation.set_state->sval[0], "appkey") == 0) {
+ app_key = esp_ble_mesh_provisioner_get_local_app_key(app_key_add.net_idx, app_key_add.app_idx);
+ if (app_key == NULL) {
+ ESP_LOGE(TAG, "CnfClient:AddAppkey,Fail,app key or network key NULL");
+ return ESP_FAIL;
+ } else {
+ memcpy(app_key_add.app_key, app_key, 16);
+ }
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&app_key_add);
+ } else if (strcmp(configuration_client_model_operation.set_state->sval[0], "appbind") == 0) {
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&mod_app_bind);
+ } else if (strcmp(configuration_client_model_operation.set_state->sval[0], "ttl") == 0) {
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&ttl_set);
+ } else if (strcmp(configuration_client_model_operation.set_state->sval[0], "proxy") == 0) {
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&proxy_set);
+ } else if (strcmp(configuration_client_model_operation.set_state->sval[0], "subadd") == 0) {
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&mod_sub_add);
+ } else if (strcmp(configuration_client_model_operation.set_state->sval[0], "subdel") == 0) {
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&mod_sub_del);
+ } else if (strcmp(configuration_client_model_operation.set_state->sval[0], "relay") == 0) {
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&relay_set);
+ } else if (strcmp(configuration_client_model_operation.set_state->sval[0], "pubset") == 0) {
+ err = esp_ble_mesh_config_client_set_state(&client_common, (esp_ble_mesh_cfg_client_set_state_t *)&mod_pub_set);
+ }
+ }
+ } else if (strcmp(configuration_client_model_operation.action_type->sval[0], "reg") == 0) {
+ err = esp_ble_mesh_register_config_client_callback(ble_mesh_configuration_client_model_cb);
+ }
+ }
+
+ if (err == ESP_OK) {
+ ESP_LOGI(TAG, "ConfigClient:OK");
+ } else {
+ ESP_LOGI(TAG, "ConfigClient:Fail");
+ }
+
+ ESP_LOGD(TAG, "exit %s %d\n", __func__, err);
+ return err;
+}
+
+
+void ble_mesh_register_configuration_client_model_command()
+{
+ configuration_client_model_operation.action_type = arg_str1("z", NULL, "", "action type");
+ configuration_client_model_operation.set_state = arg_str0("x", NULL, "", "set state");
+ configuration_client_model_operation.opcode = arg_int0("o", NULL, "", "message opcode");
+ configuration_client_model_operation.unicast_address = arg_int0("u", NULL, "