1f4854a5eSMatthias Ringwald /* 2f4854a5eSMatthias Ringwald * Copyright (C) 2018 BlueKitchen GmbH 3f4854a5eSMatthias Ringwald * 4f4854a5eSMatthias Ringwald * Redistribution and use in source and binary forms, with or without 5f4854a5eSMatthias Ringwald * modification, are permitted provided that the following conditions 6f4854a5eSMatthias Ringwald * are met: 7f4854a5eSMatthias Ringwald * 8f4854a5eSMatthias Ringwald * 1. Redistributions of source code must retain the above copyright 9f4854a5eSMatthias Ringwald * notice, this list of conditions and the following disclaimer. 10f4854a5eSMatthias Ringwald * 2. Redistributions in binary form must reproduce the above copyright 11f4854a5eSMatthias Ringwald * notice, this list of conditions and the following disclaimer in the 12f4854a5eSMatthias Ringwald * documentation and/or other materials provided with the distribution. 13f4854a5eSMatthias Ringwald * 3. Neither the name of the copyright holders nor the names of 14f4854a5eSMatthias Ringwald * contributors may be used to endorse or promote products derived 15f4854a5eSMatthias Ringwald * from this software without specific prior written permission. 16f4854a5eSMatthias Ringwald * 4. Any redistribution, use, or modification is done solely for 17f4854a5eSMatthias Ringwald * personal benefit and not for any commercial purpose or for 18f4854a5eSMatthias Ringwald * monetary gain. 19f4854a5eSMatthias Ringwald * 20f4854a5eSMatthias Ringwald * THIS SOFTWARE IS PROVIDED BY BLUEKITCHEN GMBH AND CONTRIBUTORS 21f4854a5eSMatthias Ringwald * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 22f4854a5eSMatthias Ringwald * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 23f4854a5eSMatthias Ringwald * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL MATTHIAS 24f4854a5eSMatthias Ringwald * RINGWALD OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 25f4854a5eSMatthias Ringwald * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 26f4854a5eSMatthias Ringwald * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 27f4854a5eSMatthias Ringwald * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 28f4854a5eSMatthias Ringwald * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 29f4854a5eSMatthias Ringwald * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 30f4854a5eSMatthias Ringwald * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31f4854a5eSMatthias Ringwald * SUCH DAMAGE. 32f4854a5eSMatthias Ringwald * 33f4854a5eSMatthias Ringwald * Please inquire about commercial licensing options at 34f4854a5eSMatthias Ringwald * [email protected] 35f4854a5eSMatthias Ringwald * 36f4854a5eSMatthias Ringwald */ 37f4854a5eSMatthias Ringwald 38f4854a5eSMatthias Ringwald #ifndef __MESH_ACCESS_H 39f4854a5eSMatthias Ringwald #define __MESH_ACCESS_H 40f4854a5eSMatthias Ringwald 41f4854a5eSMatthias Ringwald #include <stdint.h> 42f4854a5eSMatthias Ringwald #include <stdarg.h> 43f4854a5eSMatthias Ringwald 44*edc441d2SMatthias Ringwald #include "btstack_bool.h" 45f4854a5eSMatthias Ringwald #include "bluetooth_company_id.h" 46f4854a5eSMatthias Ringwald #include "btstack_linked_list.h" 47f4854a5eSMatthias Ringwald 48f4854a5eSMatthias Ringwald #include "mesh/mesh_lower_transport.h" 49f4854a5eSMatthias Ringwald #include "mesh/mesh_keys.h" 50f4854a5eSMatthias Ringwald #include "mesh/mesh_node.h" 51f4854a5eSMatthias Ringwald 52f4854a5eSMatthias Ringwald #ifdef __cplusplus 53f4854a5eSMatthias Ringwald extern "C" 54f4854a5eSMatthias Ringwald { 55f4854a5eSMatthias Ringwald #endif 56f4854a5eSMatthias Ringwald 575f845740SMatthias Ringwald #define MESH_SEQUENCE_NUMBER_STORAGE_INTERVAL 1000 585f845740SMatthias Ringwald 59f4854a5eSMatthias Ringwald typedef enum { 60f4854a5eSMatthias Ringwald MESH_DEFAULT_TRANSITION_STEP_RESOLUTION_100ms = 0x00u, 61f4854a5eSMatthias Ringwald MESH_DEFAULT_TRANSITION_STEP_RESOLUTION_1s, 62f4854a5eSMatthias Ringwald MESH_DEFAULT_TRANSITION_STEP_RESOLUTION_10s, 63f4854a5eSMatthias Ringwald MESH_DEFAULT_TRANSITION_STEP_RESOLUTION_10min 64f4854a5eSMatthias Ringwald } mesh_default_transition_step_resolution_t; 65f4854a5eSMatthias Ringwald 66f4854a5eSMatthias Ringwald typedef enum { 67f4854a5eSMatthias Ringwald MODEL_STATE_UPDATE_REASON_SET = 0x00u, 68f4854a5eSMatthias Ringwald MODEL_STATE_UPDATE_REASON_TRANSITION_START, 69794ab91cSMatthias Ringwald MODEL_STATE_UPDATE_REASON_TRANSITION_ACTIVE, 70f4854a5eSMatthias Ringwald MODEL_STATE_UPDATE_REASON_TRANSITION_END, 71f4854a5eSMatthias Ringwald MODEL_STATE_UPDATE_REASON_TRANSITION_ABORT, 72f4854a5eSMatthias Ringwald // MODEL_STATE_UPDATE_REASON_BOUND_STATE, 73f4854a5eSMatthias Ringwald MODEL_STATE_UPDATE_REASON_APPLICATION_CHANGE 74f4854a5eSMatthias Ringwald } model_state_update_reason_t; 75f4854a5eSMatthias Ringwald 76f4854a5eSMatthias Ringwald typedef enum { 77f4854a5eSMatthias Ringwald TRANSITION_START, 78f4854a5eSMatthias Ringwald TRANSITION_UPDATE 79f4854a5eSMatthias Ringwald } transition_event_t; 80f4854a5eSMatthias Ringwald 81f4854a5eSMatthias Ringwald typedef enum { 82f4854a5eSMatthias Ringwald MESH_TRANSITION_STATE_IDLE, 83f4854a5eSMatthias Ringwald MESH_TRANSITION_STATE_DELAYED, 84f4854a5eSMatthias Ringwald MESH_TRANSITION_STATE_ACTIVE 85f4854a5eSMatthias Ringwald } mesh_transition_state_t; 86f4854a5eSMatthias Ringwald 87f4854a5eSMatthias Ringwald typedef enum { 88f4854a5eSMatthias Ringwald MODEL_STATE_ID_GENERIC_ON_OFF = (BLUETOOTH_COMPANY_ID_BLUETOOTH_SIG_INC << 16) | 0u, 89f4854a5eSMatthias Ringwald MODEL_STATE_ID_GENERIC_LEVEL = (BLUETOOTH_COMPANY_ID_BLUETOOTH_SIG_INC << 16) | 1u, 90f4854a5eSMatthias Ringwald } model_state_id_t; 91f4854a5eSMatthias Ringwald 9263ad3643SMatthias Ringwald #define MESH_MAX_NUM_FAULTS 5 93f4854a5eSMatthias Ringwald 94b66ef5c9SMatthias Ringwald #define MESH_TRANSITION_NUM_STEPS_INFINITE 0x3f 95b66ef5c9SMatthias Ringwald 96f4854a5eSMatthias Ringwald typedef struct { 97f4854a5eSMatthias Ringwald // linked list item 98f4854a5eSMatthias Ringwald btstack_linked_item_t item; 99f4854a5eSMatthias Ringwald uint8_t test_id; 100f4854a5eSMatthias Ringwald uint16_t company_id; 10163ad3643SMatthias Ringwald uint16_t num_current_faults; 10263ad3643SMatthias Ringwald uint16_t num_registered_faults; 10363ad3643SMatthias Ringwald uint8_t current_faults[MESH_MAX_NUM_FAULTS]; 10463ad3643SMatthias Ringwald uint8_t registered_faults[MESH_MAX_NUM_FAULTS]; 10563ad3643SMatthias Ringwald } mesh_health_fault_t; 1064439045cSMatthias Ringwald 10763ad3643SMatthias Ringwald typedef struct { 10863ad3643SMatthias Ringwald // linked list of mesh_health_fault items 10963ad3643SMatthias Ringwald btstack_linked_list_t faults; 1104439045cSMatthias Ringwald uint8_t fast_period_divisor; 1114439045cSMatthias Ringwald } mesh_health_state_t; 1124439045cSMatthias Ringwald 1134439045cSMatthias Ringwald typedef struct { 114f4854a5eSMatthias Ringwald uint32_t opcode; 115f4854a5eSMatthias Ringwald uint8_t * data; 116f4854a5eSMatthias Ringwald uint16_t len; 117f4854a5eSMatthias Ringwald } mesh_access_parser_state_t; 118f4854a5eSMatthias Ringwald 119f4854a5eSMatthias Ringwald typedef struct { 120f4854a5eSMatthias Ringwald uint32_t opcode; 121f4854a5eSMatthias Ringwald const char * format; 122f4854a5eSMatthias Ringwald } mesh_access_message_t; 123f4854a5eSMatthias Ringwald 124f4854a5eSMatthias Ringwald typedef enum { 125f4854a5eSMatthias Ringwald MESH_TRANSACTION_STATUS_NEW = 0, 126f4854a5eSMatthias Ringwald MESH_TRANSACTION_STATUS_RETRANSMISSION, 127f4854a5eSMatthias Ringwald MESH_TRANSACTION_STATUS_DIFFERENT_DST_OR_SRC 128f4854a5eSMatthias Ringwald } mesh_transaction_status_t; 129f4854a5eSMatthias Ringwald 130f4854a5eSMatthias Ringwald typedef struct mesh_transition { 131649d4025SMatthias Ringwald btstack_timer_source_t timer; 132f4854a5eSMatthias Ringwald 133f4854a5eSMatthias Ringwald mesh_transition_state_t state; 134f4854a5eSMatthias Ringwald 135f4854a5eSMatthias Ringwald uint8_t transaction_identifier; 136f4854a5eSMatthias Ringwald uint32_t transaction_timestamp_ms; 137f4854a5eSMatthias Ringwald uint16_t src_address; 138f4854a5eSMatthias Ringwald uint16_t dst_address; 139f4854a5eSMatthias Ringwald 14043e34390SMatthias Ringwald uint8_t num_steps; 141332b5256SMatthias Ringwald mesh_default_transition_step_resolution_t step_resolution; 142d975617dSMatthias Ringwald uint32_t step_duration_ms; 143332b5256SMatthias Ringwald 144f4854a5eSMatthias Ringwald // to send events and/or publish changes 145f4854a5eSMatthias Ringwald mesh_model_t * mesh_model; 146f4854a5eSMatthias Ringwald 147f4854a5eSMatthias Ringwald // to execute transition 148332b5256SMatthias Ringwald void (* transition_callback)(struct mesh_transition * transition, model_state_update_reason_t event); 149f4854a5eSMatthias Ringwald } mesh_transition_t; 150f4854a5eSMatthias Ringwald 151f4854a5eSMatthias Ringwald /** 152f4854a5eSMatthias Ringwald * @brief Init access layer 153f4854a5eSMatthias Ringwald */ 154f4854a5eSMatthias Ringwald void mesh_access_init(void); 155f4854a5eSMatthias Ringwald 156f4854a5eSMatthias Ringwald /** 157f4854a5eSMatthias Ringwald * @brief Inform access layer that access message was processed by higher layer 158f4854a5eSMatthias Ringwald * @param pdu 159f4854a5eSMatthias Ringwald */ 160f4854a5eSMatthias Ringwald void mesh_access_message_processed(mesh_pdu_t * pdu); 161f4854a5eSMatthias Ringwald 162f4854a5eSMatthias Ringwald /** 163f4854a5eSMatthias Ringwald * @brief Get number of retransmissions used by default 164f4854a5eSMatthias Ringwald */ 165f4854a5eSMatthias Ringwald uint8_t mesh_access_acknowledged_message_retransmissions(void); 166f4854a5eSMatthias Ringwald 167f4854a5eSMatthias Ringwald /** 168f4854a5eSMatthias Ringwald * @brief Get retransmission timeout 169f4854a5eSMatthias Ringwald */ 170f4854a5eSMatthias Ringwald uint32_t mesh_access_acknowledged_message_timeout_ms(void); 171f4854a5eSMatthias Ringwald 172f4854a5eSMatthias Ringwald /** 173f4854a5eSMatthias Ringwald * @brief Send unacknowledged message 174f4854a5eSMatthias Ringwald * @param pdu 175f4854a5eSMatthias Ringwald */ 176f4854a5eSMatthias Ringwald void mesh_access_send_unacknowledged_pdu(mesh_pdu_t * pdu); 177f4854a5eSMatthias Ringwald 178f4854a5eSMatthias Ringwald /** 179f4854a5eSMatthias Ringwald * @brief Send acknowledged message. Retransmits message if no acknowledgement with expected opcode is received 180f4854a5eSMatthias Ringwald * @param pdu 181f4854a5eSMatthias Ringwald * @param retransmissions 182f4854a5eSMatthias Ringwald * @param ack_opcode opcode of acknowledgement 183f4854a5eSMatthias Ringwald */ 184f4854a5eSMatthias Ringwald void mesh_access_send_acknowledged_pdu(mesh_pdu_t * pdu, uint8_t retransmissions, uint32_t ack_opcode); 185f4854a5eSMatthias Ringwald 186332b5256SMatthias Ringwald 187332b5256SMatthias Ringwald // Mesh Model Transitions 1888245767eSMatthias Ringwald uint32_t mesh_access_transitions_step_ms_from_gdtt(uint8_t time_gdtt); 189f4854a5eSMatthias Ringwald uint8_t mesh_access_transitions_num_steps_from_gdtt(uint8_t time_gdtt); 190f4854a5eSMatthias Ringwald uint32_t mesh_access_time_gdtt2ms(uint8_t time_gdtt); 191332b5256SMatthias Ringwald void mesh_access_transitions_init_transaction(mesh_transition_t * transition, uint8_t transaction_identifier, uint16_t src_address, uint16_t dst_address); 192332b5256SMatthias Ringwald void mesh_access_transition_setup(mesh_model_t *mesh_model, mesh_transition_t * base_transition, uint8_t transition_time_gdtt, uint8_t delay_time_gdtt, void (*transition_callback)(mesh_transition_t * base_transition, model_state_update_reason_t event)); 193332b5256SMatthias Ringwald void mesh_access_transitions_abort_transaction(mesh_transition_t * transition); 194332b5256SMatthias Ringwald uint8_t mesh_access_transactions_get_next_transaction_id(void); 195332b5256SMatthias Ringwald mesh_transaction_status_t mesh_access_transitions_transaction_status(mesh_transition_t * transition, uint8_t transaction_identifier, uint16_t src_address, uint16_t dst_address); 196f4854a5eSMatthias Ringwald 1978ef657c6SMatthias Ringwald void mesh_access_emit_state_update_bool(btstack_packet_handler_t event_handler, uint8_t element_index, uint32_t model_identifier, 198f4854a5eSMatthias Ringwald model_state_id_t state_identifier, model_state_update_reason_t reason, uint8_t value); 199332b5256SMatthias Ringwald 2008ef657c6SMatthias Ringwald void mesh_access_emit_state_update_int16(btstack_packet_handler_t event_handler, uint8_t element_index, uint32_t model_identifier, 201f4854a5eSMatthias Ringwald model_state_id_t state_identifier, model_state_update_reason_t reason, int16_t value); 202f4854a5eSMatthias Ringwald 203f4854a5eSMatthias Ringwald // Mesh Model Publicaation 204f4854a5eSMatthias Ringwald 205f4854a5eSMatthias Ringwald /** 206f4854a5eSMatthias Ringwald * Inform Mesh Access that the state of a model has changed. may trigger state publication 207f4854a5eSMatthias Ringwald * @param mesh_model 208f4854a5eSMatthias Ringwald */ 209f4854a5eSMatthias Ringwald void mesh_access_state_changed(mesh_model_t * mesh_model); 210f4854a5eSMatthias Ringwald 211f4854a5eSMatthias Ringwald /** 212f4854a5eSMatthias Ringwald * Start Model Publication 213f4854a5eSMatthias Ringwald * @param mesh_model 214f4854a5eSMatthias Ringwald */ 215f4854a5eSMatthias Ringwald void mesh_model_publication_start(mesh_model_t * mesh_model); 216f4854a5eSMatthias Ringwald 217f4854a5eSMatthias Ringwald /** 218f4854a5eSMatthias Ringwald * Stop Model Publication 219f4854a5eSMatthias Ringwald * @param mesh_model 220f4854a5eSMatthias Ringwald */ 221f4854a5eSMatthias Ringwald void mesh_model_publication_stop(mesh_model_t * mesh_model); 222f4854a5eSMatthias Ringwald 223f4854a5eSMatthias Ringwald // Mesh PDU Getter 2245abfa389SMatthias Ringwald uint16_t mesh_pdu_ctl(mesh_pdu_t * pdu); 2253efa06c6SMatthias Ringwald uint16_t mesh_pdu_ttl(mesh_pdu_t * pdu); 226f4854a5eSMatthias Ringwald uint16_t mesh_pdu_src(mesh_pdu_t * pdu); 227f4854a5eSMatthias Ringwald uint16_t mesh_pdu_dst(mesh_pdu_t * pdu); 228f4854a5eSMatthias Ringwald uint16_t mesh_pdu_netkey_index(mesh_pdu_t * pdu); 229f4854a5eSMatthias Ringwald uint16_t mesh_pdu_appkey_index(mesh_pdu_t * pdu); 230f4854a5eSMatthias Ringwald uint16_t mesh_pdu_len(mesh_pdu_t * pdu); 231f4854a5eSMatthias Ringwald uint8_t * mesh_pdu_data(mesh_pdu_t * pdu); 232e9292fe8SMatthias Ringwald uint8_t mesh_pdu_control_opcode(mesh_pdu_t * pdu); 233f4854a5eSMatthias Ringwald 234f4854a5eSMatthias Ringwald // Mesh Access Parser 235f4854a5eSMatthias Ringwald int mesh_access_pdu_get_opcode(mesh_pdu_t * pdu, uint32_t * opcode, uint16_t * opcode_size); 236f4854a5eSMatthias Ringwald int mesh_access_parser_init(mesh_access_parser_state_t * state, mesh_pdu_t * pdu); 237f4854a5eSMatthias Ringwald void mesh_access_parser_skip(mesh_access_parser_state_t * state, uint16_t bytes_to_skip); 238f4854a5eSMatthias Ringwald uint16_t mesh_access_parser_available(mesh_access_parser_state_t * state); 239f4854a5eSMatthias Ringwald uint8_t mesh_access_parser_get_u8(mesh_access_parser_state_t * state); 240f4854a5eSMatthias Ringwald uint16_t mesh_access_parser_get_u16(mesh_access_parser_state_t * state); 241f4854a5eSMatthias Ringwald uint32_t mesh_access_parser_get_u24(mesh_access_parser_state_t * state); 242f4854a5eSMatthias Ringwald uint32_t mesh_access_parser_get_u32(mesh_access_parser_state_t * state); 243f4854a5eSMatthias Ringwald void mesh_access_parser_get_u128(mesh_access_parser_state_t * state, uint8_t * dest); 244f4854a5eSMatthias Ringwald void mesh_access_parser_get_label_uuid(mesh_access_parser_state_t * state, uint8_t * dest); 245f4854a5eSMatthias Ringwald void mesh_access_parser_get_key(mesh_access_parser_state_t * state, uint8_t * dest); 246f4854a5eSMatthias Ringwald uint32_t mesh_access_parser_get_model_identifier(mesh_access_parser_state_t * parser); 247f4854a5eSMatthias Ringwald 248bacd6cd3SMilanka Ringwald uint32_t mesh_access_parser_get_sig_model_identifier(mesh_access_parser_state_t * parser); 249bacd6cd3SMilanka Ringwald uint32_t mesh_access_parser_get_vendor_model_identifier(mesh_access_parser_state_t * parser); 250bacd6cd3SMilanka Ringwald 2516f13b0f4SMatthias Ringwald // message builder 252*edc441d2SMatthias Ringwald void mesh_access_message_free(mesh_upper_transport_pdu_t * upper); 2530323b889SMatthias Ringwald mesh_upper_transport_pdu_t * mesh_access_message_init(uint32_t opcode); 254*edc441d2SMatthias Ringwald bool mesh_access_message_add_data(mesh_upper_transport_pdu_t * pdu, const uint8_t * data, uint16_t data_len); 255*edc441d2SMatthias Ringwald bool mesh_access_message_add_uint8(mesh_upper_transport_pdu_t * pdu, uint8_t value); 256*edc441d2SMatthias Ringwald bool mesh_access_message_add_uint16(mesh_upper_transport_pdu_t * pdu, uint16_t value); 257*edc441d2SMatthias Ringwald bool mesh_access_message_add_uint24(mesh_upper_transport_pdu_t * pdu, uint16_t value); 258*edc441d2SMatthias Ringwald bool mesh_access_message_add_uint32(mesh_upper_transport_pdu_t * pdu, uint16_t value); 259*edc441d2SMatthias Ringwald bool mesh_access_message_add_model_identifier(mesh_upper_transport_pdu_t * pdu, uint32_t model_identifier); 260e41a4a08SMatthias Ringwald 261f4854a5eSMatthias Ringwald // message builder using template 2620323b889SMatthias Ringwald mesh_upper_transport_pdu_t * mesh_access_setup_message(const mesh_access_message_t *message_template, ...); 263f4854a5eSMatthias Ringwald 264f4854a5eSMatthias Ringwald #ifdef __cplusplus 265f4854a5eSMatthias Ringwald } /* end of extern "C" */ 266f4854a5eSMatthias Ringwald #endif 267f4854a5eSMatthias Ringwald 268f4854a5eSMatthias Ringwald #endif 269