2021-03-11 14:48:59 +01:00

245 lines
8.6 KiB
C

/*
* ESPRESSIF MIT License
*
* Copyright (c) 2018 <ESPRESSIF SYSTEMS (SHANGHAI) PTE LTD>
*
* Permission is hereby granted for use on ESPRESSIF SYSTEMS products only, in which case,
* it is free of charge, to any person obtaining a copy of this software and associated
* documentation files (the "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the Software is furnished
* to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all copies or
* substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*
*/
/* HomeKit Smart Outlet Example
*/
#include <stdio.h>
#include <string.h>
#include <freertos/FreeRTOS.h>
#include <freertos/task.h>
#include <freertos/queue.h>
#include <esp_log.h>
#include <driver/gpio.h>
#include <hap.h>
#include <hap_apple_servs.h>
#include <hap_apple_chars.h>
//#include <app_wifi.h>
//#include <app_hap_setup_payload.h>
static const char *TAG = "HAP outlet";
#define SMART_OUTLET_TASK_PRIORITY 1
#define SMART_OUTLET_TASK_STACKSIZE 4 * 1024
#define SMART_OUTLET_TASK_NAME "hap_outlet"
//#define OUTLET_IN_USE_GPIO GPIO_NUM_0
#define OUTLET_IN_USE_GPIO -1
#define ESP_INTR_FLAG_DEFAULT 0
static xQueueHandle s_esp_evt_queue = NULL;
/**
* @brief the recover outlet in use gpio interrupt function
*/
static void IRAM_ATTR outlet_in_use_isr(void* arg)
{
uint32_t gpio_num = (uint32_t) arg;
xQueueSendFromISR(s_esp_evt_queue, &gpio_num, NULL);
}
/**
* Enable a GPIO Pin for Outlet in Use Detection
*/
static void outlet_in_use_key_init(uint32_t key_gpio_pin)
{
gpio_config_t io_conf;
/* Interrupt for both the edges */
io_conf.intr_type = GPIO_INTR_ANYEDGE;
/* Bit mask of the pins */
io_conf.pin_bit_mask = 1 << key_gpio_pin;
/* Set as input mode */
io_conf.mode = GPIO_MODE_INPUT;
/* Enable internal pull-up */
io_conf.pull_up_en = 1;
/* Disable internal pull-down */
io_conf.pull_down_en = 0;
/* Set the GPIO configuration */
gpio_config(&io_conf);
/* Install gpio isr service */
gpio_install_isr_service(ESP_INTR_FLAG_DEFAULT);
/* Hook isr handler for specified gpio pin */
gpio_isr_handler_add(key_gpio_pin, outlet_in_use_isr, (void*)key_gpio_pin);
}
/**
* Initialize the Smart Outlet Hardware.Here, we just enebale the Outlet-In-Use detection.
*/
void smart_outlet_hardware_init(gpio_num_t gpio_num)
{
s_esp_evt_queue = xQueueCreate(2, sizeof(uint32_t));
if (s_esp_evt_queue != NULL) {
outlet_in_use_key_init(gpio_num);
}
}
/* Mandatory identify routine for the accessory.
* In a real accessory, something like LED blink should be implemented
* got visual identification
*/
static int outlet_identify(hap_acc_t *ha)
{
ESP_LOGI(TAG, "Accessory identified");
return HAP_SUCCESS;
}
/* A dummy callback for handling a write on the "On" characteristic of Outlet.
* In an actual accessory, this should control the hardware
*/
static int outlet_write(hap_write_data_t write_data[], int count,
void *serv_priv, void *write_priv)
{
int i, ret = HAP_SUCCESS;
hap_write_data_t *write;
for (i = 0; i < count; i++) {
write = &write_data[i];
if (!strcmp(hap_char_get_type_uuid(write->hc), HAP_CHAR_UUID_ON)) {
//ESP_LOGI(TAG, "Received Write. Outlet %s", write->val.b ? "On" : "Off");
ESP_LOG_LEVEL(ESP_LOG_INFO, TAG, "Received Write. Outlet %s", write->val.b ? "On" : "Off");
/* TODO: Control Actual Hardware */
hap_char_update_val(write->hc, &(write->val));
*(write->status) = HAP_STATUS_SUCCESS;
} else {
*(write->status) = HAP_STATUS_RES_ABSENT;
}
}
return ret;
}
/*The main thread for handling the Smart Outlet Accessory */
static void smart_outlet_thread_entry(void *p)
{
hap_acc_t *accessory;
hap_serv_t *service;
/* Initialize the HAP core */
hap_init(HAP_TRANSPORT_WIFI);
/* Initialise the mandatory parameters for Accessory which will be added as
* the mandatory services internally
*/
hap_acc_cfg_t cfg = {
.name = "Esp-Smart-Outlet",
.manufacturer = "Espressif",
.model = "EspSmartOutlet01",
.serial_num = "001122334455",
.fw_rev = "0.9.0",
.hw_rev = NULL,
.pv = "1.1.0",
.identify_routine = outlet_identify,
.cid = HAP_CID_OUTLET,
};
/* Create accessory object */
accessory = hap_acc_create(&cfg);
/* Add a dummy Product Data */
uint8_t product_data[] = {'E','S','P','3','2','H','A','P'};
hap_acc_add_product_data(accessory, product_data, sizeof(product_data));
/* Create the Outlet Service. Include the "name" since this is a user visible service */
service = hap_serv_outlet_create(false, false);
hap_serv_add_char(service, hap_char_name_create("My Smart Outlet"));
/* Get pointer to the outlet in use characteristic which we need to monitor for state changes */
hap_char_t *outlet_in_use = hap_serv_get_char_by_uuid(service, HAP_CHAR_UUID_OUTLET_IN_USE);
/* Set the write callback for the service */
hap_serv_set_write_cb(service, outlet_write);
/* Add the Outlet Service to the Accessory Object */
hap_acc_add_serv(accessory, service);
/* Add the Accessory to the HomeKit Database */
hap_add_accessory(accessory);
/* Initialize the appliance specific hardware. This enables out-in-use detection */
smart_outlet_hardware_init(OUTLET_IN_USE_GPIO);
/* For production accessories, the setup code shouldn't be programmed on to
* the device. Instead, the setup info, derived from the setup code must
* be used. Use the factory_nvs_gen utility to generate this data and then
* flash it into the factory NVS partition.
*
* By default, the setup ID and setup info will be read from the factory_nvs
* Flash partition and so, is not required to set here explicitly.
*
* However, for testing purpose, this can be overridden by using hap_set_setup_code()
* and hap_set_setup_id() APIs, as has been done here.
*/
hap_set_setup_code("111-11-111");
hap_set_setup_id("ES32");
#ifdef CONFIG_EXAMPLE_USE_HARDCODED_SETUP_CODE
/* Unique Setup code of the format xxx-xx-xxx. Default: 111-22-333 */
hap_set_setup_code(CONFIG_EXAMPLE_SETUP_CODE);
/* Unique four character Setup Id. Default: ES32 */
hap_set_setup_id(CONFIG_EXAMPLE_SETUP_ID);
#ifdef CONFIG_APP_WIFI_USE_WAC_PROVISIONING
app_hap_setup_payload(CONFIG_EXAMPLE_SETUP_CODE, CONFIG_EXAMPLE_SETUP_ID, true, cfg.cid);
#else
app_hap_setup_payload(CONFIG_EXAMPLE_SETUP_CODE, CONFIG_EXAMPLE_SETUP_ID, false, cfg.cid);
#endif
#endif
/* Enable Hardware MFi authentication (applicable only for MFi variant of SDK) */
hap_enable_mfi_auth(HAP_MFI_AUTH_HW);
/* Initialize Wi-Fi */
//app_wifi_init();
/* After all the initializations are done, start the HAP core */
hap_start();
/* Start Wi-Fi */
//app_wifi_start(portMAX_DELAY);
uint32_t io_num = OUTLET_IN_USE_GPIO;
hap_val_t appliance_value = {
.b = true,
};
/* Listen for Outlet-In-Use state change events. Other read/write functionality will be handled
* by the HAP Core.
* When the Outlet in Use GPIO goes low, it means Outlet is not in use.
* When the Outlet in Use GPIO goes high, it means Outlet is in use.
* Applications can define own logic as per their hardware.
*/
while (1) {
if (xQueueReceive(s_esp_evt_queue, &io_num, portMAX_DELAY) == pdFALSE) {
ESP_LOGI(TAG, "Outlet-In-Use trigger FAIL");
} else {
appliance_value.b = gpio_get_level(io_num);
/* If any state change is detected, update the Outlet In Use characteristic value */
hap_char_update_val(outlet_in_use, &appliance_value);
ESP_LOGI(TAG, "Outlet-In-Use triggered [%d]", appliance_value.b);
}
}
}
void homekit_main()
{
/* Create the application thread */
xTaskCreate(smart_outlet_thread_entry, SMART_OUTLET_TASK_NAME, SMART_OUTLET_TASK_STACKSIZE,
NULL, SMART_OUTLET_TASK_PRIORITY, NULL);
}