1d4afb5ceSopenharmony_ci/* 2d4afb5ceSopenharmony_ci * Generic GPIO led 3d4afb5ceSopenharmony_ci * 4d4afb5ceSopenharmony_ci * Copyright (C) 2019 - 2020 Andy Green <andy@warmcat.com> 5d4afb5ceSopenharmony_ci * 6d4afb5ceSopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining a copy 7d4afb5ceSopenharmony_ci * of this software and associated documentation files (the "Software"), to 8d4afb5ceSopenharmony_ci * deal in the Software without restriction, including without limitation the 9d4afb5ceSopenharmony_ci * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or 10d4afb5ceSopenharmony_ci * sell copies of the Software, and to permit persons to whom the Software is 11d4afb5ceSopenharmony_ci * furnished to do so, subject to the following conditions: 12d4afb5ceSopenharmony_ci * 13d4afb5ceSopenharmony_ci * The above copyright notice and this permission notice shall be included in 14d4afb5ceSopenharmony_ci * all copies or substantial portions of the Software. 15d4afb5ceSopenharmony_ci * 16d4afb5ceSopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 17d4afb5ceSopenharmony_ci * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 18d4afb5ceSopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 19d4afb5ceSopenharmony_ci * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 20d4afb5ceSopenharmony_ci * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 21d4afb5ceSopenharmony_ci * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS 22d4afb5ceSopenharmony_ci * IN THE SOFTWARE. 23d4afb5ceSopenharmony_ci */ 24d4afb5ceSopenharmony_ci#include "private-lib-core.h" 25d4afb5ceSopenharmony_ci 26d4afb5ceSopenharmony_ci#include "drivers/led/private-lib-drivers-led.h" 27d4afb5ceSopenharmony_ci 28d4afb5ceSopenharmony_ci/* 29d4afb5ceSopenharmony_ci * 64 entry interpolated CIE correction 30d4afb5ceSopenharmony_ci * https://en.wikipedia.org/wiki/Lightness 31d4afb5ceSopenharmony_ci */ 32d4afb5ceSopenharmony_ci 33d4afb5ceSopenharmony_ciuint16_t cie[] = { 34d4afb5ceSopenharmony_ci 0, 113, 227, 340, 454, 568, 688, 824, 976, 1146, 35d4afb5ceSopenharmony_ci 1335, 1543, 1772, 2023, 2296, 2592, 2914, 3260, 3633, 4034, 36d4afb5ceSopenharmony_ci 4463, 4921, 5409, 5929, 6482, 7067, 7687, 8341, 9032, 9761, 37d4afb5ceSopenharmony_ci 10527, 11332, 12178, 13064, 13993, 14964, 15980, 17040, 18146, 19299, 38d4afb5ceSopenharmony_ci 20500, 21750, 23049, 24400, 25802, 27256, 28765, 30328, 31946, 33622, 39d4afb5ceSopenharmony_ci 35354, 37146, 38996, 40908, 42881, 44916, 47014, 49177, 51406, 53700, 40d4afb5ceSopenharmony_ci 56062, 58492, 60992, 63561, 41d4afb5ceSopenharmony_ci 65535 /* for interpolation */ 42d4afb5ceSopenharmony_ci}; 43d4afb5ceSopenharmony_ci 44d4afb5ceSopenharmony_ci/* 45d4afb5ceSopenharmony_ci * This is the default intensity correction function, it can be overridden 46d4afb5ceSopenharmony_ci * per-led to eg, normalize intensity of different leds 47d4afb5ceSopenharmony_ci */ 48d4afb5ceSopenharmony_ci 49d4afb5ceSopenharmony_cistatic lws_led_intensity_t 50d4afb5ceSopenharmony_cicie_antilog(lws_led_intensity_t lin) 51d4afb5ceSopenharmony_ci{ 52d4afb5ceSopenharmony_ci return (cie[lin >> 10] * (0x3ff - (lin & 0x3ff)) + 53d4afb5ceSopenharmony_ci cie[(lin >> 10) + 1] * (lin & 0x3ff)) / 0x3ff; 54d4afb5ceSopenharmony_ci} 55d4afb5ceSopenharmony_ci 56d4afb5ceSopenharmony_cistatic void 57d4afb5ceSopenharmony_cilws_seq_advance(lws_led_state_t *lcs, lws_led_state_ch_t *ch) 58d4afb5ceSopenharmony_ci{ 59d4afb5ceSopenharmony_ci if (!ch->seq) 60d4afb5ceSopenharmony_ci return; 61d4afb5ceSopenharmony_ci 62d4afb5ceSopenharmony_ci if (ch->phase_budget != LWS_SEQ_LEDPHASE_TOTAL_ENDLESS && 63d4afb5ceSopenharmony_ci (ch->phase_budget < ch->step || !ch->phase_budget)) { 64d4afb5ceSopenharmony_ci 65d4afb5ceSopenharmony_ci /* we are done */ 66d4afb5ceSopenharmony_ci 67d4afb5ceSopenharmony_ci ch->seq = NULL; 68d4afb5ceSopenharmony_ci if (!(--lcs->timer_refcount)) { 69d4afb5ceSopenharmony_ci#if defined(LWS_PLAT_TIMER_STOP) 70d4afb5ceSopenharmony_ci LWS_PLAT_TIMER_STOP(lcs->timer); 71d4afb5ceSopenharmony_ci#endif 72d4afb5ceSopenharmony_ci } 73d4afb5ceSopenharmony_ci 74d4afb5ceSopenharmony_ci return; 75d4afb5ceSopenharmony_ci } 76d4afb5ceSopenharmony_ci 77d4afb5ceSopenharmony_ci ch->ph += ch->step; 78d4afb5ceSopenharmony_ci if (ch->phase_budget != LWS_SEQ_LEDPHASE_TOTAL_ENDLESS) 79d4afb5ceSopenharmony_ci ch->phase_budget -= ch->step; 80d4afb5ceSopenharmony_ci} 81d4afb5ceSopenharmony_ci 82d4afb5ceSopenharmony_cistatic lws_led_intensity_t 83d4afb5ceSopenharmony_cilws_seq_sample(const lws_led_gpio_map_t *map, lws_led_state_chs_t *chs) 84d4afb5ceSopenharmony_ci{ 85d4afb5ceSopenharmony_ci unsigned int i; 86d4afb5ceSopenharmony_ci 87d4afb5ceSopenharmony_ci if (chs->seqs[LLSI_CURR].seq) 88d4afb5ceSopenharmony_ci chs->seqs[LLSI_CURR].last = chs->seqs[LLSI_CURR].seq-> 89d4afb5ceSopenharmony_ci func(chs->seqs[LLSI_CURR].ph); 90d4afb5ceSopenharmony_ci 91d4afb5ceSopenharmony_ci if (chs->seqs[LLSI_TRANS].seq) { 92d4afb5ceSopenharmony_ci /* 93d4afb5ceSopenharmony_ci * If a transition is ongoing, we need to use the transition 94d4afb5ceSopenharmony_ci * intensity as the mixing factor between the still-live current 95d4afb5ceSopenharmony_ci * and newly-live next sequences 96d4afb5ceSopenharmony_ci */ 97d4afb5ceSopenharmony_ci chs->seqs[LLSI_TRANS].last = chs->seqs[LLSI_TRANS].seq-> 98d4afb5ceSopenharmony_ci func(chs->seqs[LLSI_TRANS].ph); 99d4afb5ceSopenharmony_ci 100d4afb5ceSopenharmony_ci if (chs->seqs[LLSI_NEXT].seq) 101d4afb5ceSopenharmony_ci chs->seqs[LLSI_NEXT].last = chs->seqs[LLSI_NEXT].seq-> 102d4afb5ceSopenharmony_ci func(chs->seqs[LLSI_NEXT].ph); 103d4afb5ceSopenharmony_ci 104d4afb5ceSopenharmony_ci i = (lws_led_intensity_t)((( 105d4afb5ceSopenharmony_ci (unsigned int)chs->seqs[LLSI_CURR].last * 106d4afb5ceSopenharmony_ci (65535 - chs->seqs[LLSI_TRANS].last) >> 16) + 107d4afb5ceSopenharmony_ci (((unsigned int)chs->seqs[LLSI_NEXT].last * 108d4afb5ceSopenharmony_ci (unsigned int)chs->seqs[LLSI_TRANS].last) >> 16))); 109d4afb5ceSopenharmony_ci } else 110d4afb5ceSopenharmony_ci i = chs->seqs[LLSI_CURR].last; 111d4afb5ceSopenharmony_ci 112d4afb5ceSopenharmony_ci return map->intensity_correction ? map->intensity_correction(i) : 113d4afb5ceSopenharmony_ci cie_antilog((lws_led_intensity_t)i); 114d4afb5ceSopenharmony_ci} 115d4afb5ceSopenharmony_ci 116d4afb5ceSopenharmony_civoid 117d4afb5ceSopenharmony_cilws_seq_timer_handle(lws_led_state_t *lcs) 118d4afb5ceSopenharmony_ci{ 119d4afb5ceSopenharmony_ci lws_led_gpio_controller_t *lgc = lcs->controller; 120d4afb5ceSopenharmony_ci lws_led_state_chs_t *chs = (lws_led_state_chs_t *)&lcs[1]; 121d4afb5ceSopenharmony_ci const lws_led_gpio_map_t *map = &lgc->led_map[0]; 122d4afb5ceSopenharmony_ci unsigned int n; 123d4afb5ceSopenharmony_ci 124d4afb5ceSopenharmony_ci for (n = 0; n < lgc->count_leds; n++) { 125d4afb5ceSopenharmony_ci 126d4afb5ceSopenharmony_ci lgc->led_ops.intensity(&lgc->led_ops, map->name, 127d4afb5ceSopenharmony_ci lws_seq_sample(map, chs)); 128d4afb5ceSopenharmony_ci 129d4afb5ceSopenharmony_ci lws_seq_advance(lcs, &chs->seqs[LLSI_CURR]); 130d4afb5ceSopenharmony_ci 131d4afb5ceSopenharmony_ci if (chs->seqs[LLSI_TRANS].seq) { 132d4afb5ceSopenharmony_ci lws_seq_advance(lcs, &chs->seqs[LLSI_NEXT]); 133d4afb5ceSopenharmony_ci lws_seq_advance(lcs, &chs->seqs[LLSI_TRANS]); 134d4afb5ceSopenharmony_ci 135d4afb5ceSopenharmony_ci /* 136d4afb5ceSopenharmony_ci * When we finished the transition, we can make the 137d4afb5ceSopenharmony_ci * "next" sequence the current sequence and no need for 138d4afb5ceSopenharmony_ci * a "next" or a transition any more. 139d4afb5ceSopenharmony_ci */ 140d4afb5ceSopenharmony_ci 141d4afb5ceSopenharmony_ci if (!chs->seqs[LLSI_TRANS].seq) { 142d4afb5ceSopenharmony_ci chs->seqs[LLSI_CURR] = chs->seqs[LLSI_NEXT]; 143d4afb5ceSopenharmony_ci chs->seqs[LLSI_NEXT].seq = NULL; 144d4afb5ceSopenharmony_ci } 145d4afb5ceSopenharmony_ci } 146d4afb5ceSopenharmony_ci 147d4afb5ceSopenharmony_ci map++; 148d4afb5ceSopenharmony_ci chs++; 149d4afb5ceSopenharmony_ci } 150d4afb5ceSopenharmony_ci} 151d4afb5ceSopenharmony_ci 152d4afb5ceSopenharmony_cistatic int 153d4afb5ceSopenharmony_cilws_led_set_chs_seq(struct lws_led_state *lcs, lws_led_state_ch_t *dest, 154d4afb5ceSopenharmony_ci const lws_led_sequence_def_t *def) 155d4afb5ceSopenharmony_ci{ 156d4afb5ceSopenharmony_ci int steps; 157d4afb5ceSopenharmony_ci 158d4afb5ceSopenharmony_ci dest->seq = def; 159d4afb5ceSopenharmony_ci dest->ph = def->ledphase_offset; 160d4afb5ceSopenharmony_ci dest->phase_budget = def->ledphase_total; 161d4afb5ceSopenharmony_ci 162d4afb5ceSopenharmony_ci /* 163d4afb5ceSopenharmony_ci * We need to compute the incremental phase angle step to cover the 164d4afb5ceSopenharmony_ci * total number of phases in the indicated ms, incrementing at the 165d4afb5ceSopenharmony_ci * timer rate of LWS_LED_SEQUENCER_UPDATE_RATE_HZ. Eg, 166d4afb5ceSopenharmony_ci * 167d4afb5ceSopenharmony_ci * 65536 phase steps (one cycle) in 2000ms at 30Hz timer rate means we 168d4afb5ceSopenharmony_ci * will update 2000ms / 33ms = 60 times, so we must step at at 169d4afb5ceSopenharmony_ci * 65536 / 60 = 1092 phase angle resolution 170d4afb5ceSopenharmony_ci */ 171d4afb5ceSopenharmony_ci 172d4afb5ceSopenharmony_ci steps = def->ms / LWS_LED_SEQUENCER_UPDATE_INTERVAL_MS; 173d4afb5ceSopenharmony_ci dest->step = (def->ledphase_total != LWS_SEQ_LEDPHASE_TOTAL_ENDLESS ? 174d4afb5ceSopenharmony_ci def->ledphase_total : LWS_LED_FUNC_PHASE) / (steps ? steps : 1); 175d4afb5ceSopenharmony_ci 176d4afb5ceSopenharmony_ci if (!lcs->timer_refcount++) { 177d4afb5ceSopenharmony_ci#if defined(LWS_PLAT_TIMER_START) 178d4afb5ceSopenharmony_ci LWS_PLAT_TIMER_START(lcs->timer); 179d4afb5ceSopenharmony_ci#endif 180d4afb5ceSopenharmony_ci } 181d4afb5ceSopenharmony_ci 182d4afb5ceSopenharmony_ci return steps; 183d4afb5ceSopenharmony_ci} 184d4afb5ceSopenharmony_ci 185d4afb5ceSopenharmony_ciint 186d4afb5ceSopenharmony_cilws_led_transition(struct lws_led_state *lcs, const char *name, 187d4afb5ceSopenharmony_ci const lws_led_sequence_def_t *next, 188d4afb5ceSopenharmony_ci const lws_led_sequence_def_t *trans) 189d4afb5ceSopenharmony_ci{ 190d4afb5ceSopenharmony_ci lws_led_state_chs_t *chs = (lws_led_state_chs_t *)&lcs[1]; 191d4afb5ceSopenharmony_ci int index = lws_led_gpio_lookup(&lcs->controller->led_ops, name); 192d4afb5ceSopenharmony_ci 193d4afb5ceSopenharmony_ci if (index < 0) 194d4afb5ceSopenharmony_ci return 1; 195d4afb5ceSopenharmony_ci 196d4afb5ceSopenharmony_ci lws_led_set_chs_seq(lcs, &chs[index].seqs[LLSI_TRANS], trans); 197d4afb5ceSopenharmony_ci lws_led_set_chs_seq(lcs, &chs[index].seqs[LLSI_NEXT], next); 198d4afb5ceSopenharmony_ci 199d4afb5ceSopenharmony_ci return 0; 200d4afb5ceSopenharmony_ci} 201