Re: ESP32 domácí automatizace
Pavel Brychta
pavel.brychta na duhasys.eu
Úterý Srpen 3 07:46:03 CEST 2021
Vůbec nechápu, proč se takto jednoduchý úkol snažíte komplikovat
vláknem, navíc s tak primitivně provedenou synchronizací a blokováním
(špatně ošetřeným hazardem - od toho jsou v RTOS semafory). Normálně
bych použil jeden Ticker objekt, napsal dvě funkce ventilon a ventiloff
a pomocí FSM řešil synchronizaci.
P.B.
Dne 03. 08. 21 v 7:35 Jan Půhoný napsal(a):
> Tak jsem to přepsal na vTaskDelayUntil , ale chová se to stejně.
> Problém je totiž v tom, že když se stiskne tlačítko we webovém
> rozhraní pro ovládání toho ventilu rychleji než se provede ten task,
> pustí se task pro otevírání a zavírání přes sebe a pak dochází k tomu,
> že to zůstane viset někde uprostřed. Pro tuto aplikaci asi nebude
> zásadní rozdíl mezi vTaskDelayUntil a vTaskDelay.
>
> Ošetřil jsem to přidáním proměnné unBlock, která blokuje čtení
> tlačítka po dobu spuštění tasku. Takto to funguje spolehlivě. Nicméně
> předpokládám, že to jde vyřešit nějak elegantněji. *Jak se to správně
> dělá?*
>
> Díky,
>
> HP
>
> Callback tlačítka:
> button1.attachCallback([&](bool value)
> {
> /* Print our new button value received from dashboard */
> Serial.println("Button Triggered: " + String((value) ? "true" : "false"));
> /* Make sure we update our button's value and send update to dashboard */
> digitalWrite(ledPinRed, value);
> if (unBlock)
> {
> if (value)
> {
> xTaskCreatePinnedToCore(
> vent1On, // Function that should be called
> "vent1On", // Name of the task (for debugging)
> 2000, // Stack size (bytes)
> NULL, // Parameter to pass
> 1, // Task priority
> NULL, // Task handle
> 0 // Core 1/0
> );
> }
> else
> {
> xTaskCreatePinnedToCore(
> vent1Off, // Function that should be called
> "vent1Off", // Name of the task (for debugging)
> 2000, // Stack size (bytes)
> NULL, // Parameter to pass
> 1, // Task priority
> NULL, // Task handle
> 0 // Core 1/0
> );
> }
> }
> dashboard.sendUpdates();
> });
>
> Ovládání ventilů:
> void vent1On(void *parameter)
> {
> TickType_t xLastWakeTime;
> const TickType_t xFrequency = (20000 / portTICK_PERIOD_MS);
> // Initialise the xLastWakeTime variable with the current time.
> xLastWakeTime = xTaskGetTickCount();
> for (;;)
> {
> unBlock = 0;
> digitalWrite(ventMain1, HIGH);
> digitalWrite(ventOn1, HIGH);
> digitalWrite(ledPinRed, HIGH);
> button1.update(1);
> systemCard.update((float)xFrequency);
> // Pause the task fo 20s
> vTaskDelayUntil(&xLastWakeTime, xFrequency);
> //vTaskDelay(20000 / portTICK_PERIOD_MS);
> digitalWrite(ventMain1, LOW);
> digitalWrite(ventOn1, LOW);
> digitalWrite(ledPinRed, LOW);
> unBlock = 1;
> vTaskDelete(NULL); //spustí se jen jednou
> }
> }
> void vent1Off(void *parameter)
> {
> TickType_t xLastWakeTime;
> const TickType_t xFrequency = (20000 / portTICK_PERIOD_MS);
> // Initialise the xLastWakeTime variable with the current time.
> xLastWakeTime = xTaskGetTickCount();
> for (;;)
> { // infinite loop
> unBlock = 0;
> digitalWrite(ventMain1, HIGH);
> digitalWrite(ventOn1, LOW);
> digitalWrite(ledPinRed, HIGH);
> button1.update(0);
> systemCard.update((float)xFrequency);
> // Pause the task fo 20s
> vTaskDelayUntil(&xLastWakeTime, xFrequency);
> //vTaskDelay(20000 / portTICK_PERIOD_MS);
> digitalWrite(ventMain1, LOW);
> digitalWrite(ventOn1, LOW);
> digitalWrite(ledPinRed, LOW);
> unBlock = 1;
> vTaskDelete(NULL); //spustí se jen jednou
> }
> }
>
>
>
> po 2. 8. 2021 v 16:11 odesílatel Jaroslav Buchta
> <jaroslav.buchta na hascomp.cz <mailto:jaroslav.buchta na hascomp.cz>> napsal:
>
> A zrovna nad tim badam a jedna vec mi nejde do hlavy:
> v modulu port.c je obsluha
>
> BaseType_t xPortSysTickHandler( void )
> {
> BaseType_t ret;
>
> portbenchmarkIntLatency();
> traceISR_ENTER(SYSTICK_INTR_ID);
> ret = xTaskIncrementTick();
> if( ret != pdFALSE )
> {
> portYIELD_FROM_ISR();
> } else {
> traceISR_EXIT();
> }
> return ret;
> }
>
> Je logicky umistena v IRAM sekci ale proc??? Funkce nema
> specifikovany IRAM_ATTR, ani v hlavicce.
> zrovna tak BaseType_t xTaskIncrementTick( void ) v tasks.c
>
> Tady uz treba attribut je
> void IRAM_ATTR esp_vApplicationTickHook(void)
> {
> int n;
> int core = xPortGetCoreID();
> for (n=0; n<MAX_HOOKS; n++) {
> if (tick_cb[core][n]!=NULL) {
> tick_cb[core][n]();
> }
> }
> }
>
> Funkce se vnoruji v tomto poradi.
>
> Zkousel jsem se povesit na hook od toho casovace a funguje to, jak
> z Flash, tak z IRAM (to je asi rozumnejsi, kdyz je to v ISR)
>
>
> Dne 02.08.2021 v 12:32 Jaroslav Buchta napsal(a):
>> At to ctu stale dokola, nevidim prostor pro zkraceni FreeRTOS -
>> ESP32 - — ESP-IDF Programming Guide latest documentation
>> (espressif.com)
>> <https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/system/freertos.html#task-api>
>> Jen je tam upozorneni, ze je nejista prodleva, kdy bude zavolana
>> funkce a az od te doby se to pocita. vTaskDelayUntil bude mit asi
>> jediny rozdil v tom, ze je tam dany cas ukonceni cekani jako
>> parametr, u vTaskDelay se tato hodnot asi ulozi v okamziku volani.
>> A i tak by se jednalo spis o milisekundy nez sekundy...
>>
>> Koukam na to, ze ty funkce pracuji v principu uplne stejne,
>> pouziji pro uspani
>> prvAddCurrentTaskToDelayedList(...);
>> jenom jinak pocitaji dobu pro uspani...
>>
>> Pokud opravdu muze dojit ke zkraceni intervalu, bylo by v systemu
>> neco velmi spatne. Neprobehne nahodou spis restart?
>>
>> //////////////////////////////
>>
>> void vTaskDelay( const TickType_t xTicksToDelay )
>> {
>> BaseType_t xAlreadyYielded = pdFALSE;
>>
>> /* A delay time of zero just forces a reschedule. */
>> if( xTicksToDelay > ( TickType_t ) 0U )
>> {
>> configASSERT( uxSchedulerSuspended[xPortGetCoreID()]
>> == 0 );
>> taskENTER_CRITICAL( &xTaskQueueMutex );
>> {
>> traceTASK_DELAY();
>>
>> /* A task that is removed from the event list
>> while the
>> scheduler is suspended will not get placed in the
>> ready
>> list or removed from the blocked list until the
>> scheduler
>> is resumed.
>>
>> This task cannot be in an event list as it is the
>> currently
>> executing task. */
>> prvAddCurrentTaskToDelayedList( xPortGetCoreID(),
>> xTicksToDelay );
>> }
>> taskEXIT_CRITICAL( &xTaskQueueMutex );
>> }
>> else
>> {
>> mtCOVERAGE_TEST_MARKER();
>> }
>>
>> /* Force a reschedule if xTaskResumeAll has not already
>> done so, we may
>> have put ourselves to sleep. */
>> if( xAlreadyYielded == pdFALSE )
>> {
>> portYIELD_WITHIN_API();
>> }
>> else
>> {
>> mtCOVERAGE_TEST_MARKER();
>> }
>> }
>>
>>
>>
>> void vTaskDelayUntil( TickType_t * const pxPreviousWakeTime,
>> const TickType_t xTimeIncrement )
>> {
>> TickType_t xTimeToWake;
>> BaseType_t xAlreadyYielded = pdFALSE, xShouldDelay = pdFALSE;
>>
>> configASSERT( pxPreviousWakeTime );
>> configASSERT( ( xTimeIncrement > 0U ) );
>> configASSERT( uxSchedulerSuspended[xPortGetCoreID()] == 0 );
>>
>> taskENTER_CRITICAL( &xTaskQueueMutex );
>> {
>> /* Minor optimisation. The tick count cannot change
>> in this
>> block. */
>> const TickType_t xConstTickCount = xTickCount;
>>
>> /* Generate the tick time at which the task wants to
>> wake. */
>> xTimeToWake = *pxPreviousWakeTime + xTimeIncrement;
>>
>> if( xConstTickCount < *pxPreviousWakeTime )
>> {
>> /* The tick count has overflowed since this
>> function was
>> lasted called. In this case the only time we
>> should ever
>> actually delay is if the wake time has also
>> overflowed,
>> and the wake time is greater than the tick time.
>> When this
>> is the case it is as if neither time had
>> overflowed. */
>> if( ( xTimeToWake < *pxPreviousWakeTime ) && (
>> xTimeToWake > xConstTickCount ) )
>> {
>> xShouldDelay = pdTRUE;
>> }
>> else
>> {
>> mtCOVERAGE_TEST_MARKER();
>> }
>> }
>> else
>> {
>> /* The tick time has not overflowed. In this
>> case we will
>> delay if either the wake time has overflowed,
>> and/or the
>> tick time is less than the wake time. */
>> if( ( xTimeToWake < *pxPreviousWakeTime ) || (
>> xTimeToWake > xConstTickCount ) )
>> {
>> xShouldDelay = pdTRUE;
>> }
>> else
>> {
>> mtCOVERAGE_TEST_MARKER();
>> }
>> }
>>
>> /* Update the wake time ready for the next call. */
>> *pxPreviousWakeTime = xTimeToWake;
>>
>> if( xShouldDelay != pdFALSE )
>> {
>> traceTASK_DELAY_UNTIL();
>>
>> /* prvAddCurrentTaskToDelayedList() needs the
>> block time, not
>> the time to wake, so subtract the current tick
>> count. */
>> prvAddCurrentTaskToDelayedList( xPortGetCoreID(),
>> xTimeToWake - xConstTickCount );
>> }
>> else
>> {
>> mtCOVERAGE_TEST_MARKER();
>> }
>> }
>> taskEXIT_CRITICAL( &xTaskQueueMutex );
>>
>> /* Force a reschedule if xTaskResumeAll has not already
>> done so, we may
>> have put ourselves to sleep. */
>> if( xAlreadyYielded == pdFALSE )
>> {
>> portYIELD_WITHIN_API();
>> }
>> else
>> {
>> mtCOVERAGE_TEST_MARKER();
>> }
>> }
>>
>>
>> Dne 02.08.2021 v 12:09 as5s napsal(a):
>>>
>>> Netusim ako to presne funguje, ale v dokumentacii na to upozornuju.
>>>
>>> ale je mi cudne ze sa to skrati. O kolko sa to skrati?
>>>
>>> Myre
>>>
>>>
>>> On 2. 8. 2021 10:35, Jaroslav Buchta wrote:
>>>> To jako opravdu vTaskDelay muze fungovat nespolehlive, teda
>>>> jeste k tomu se zkratit?
>>>> To snad primo pocita systemova preruseni od casovace (default
>>>> tusim 100Hz), ze by se nejaky nestihly bych jeste pochopil...
>>>>
>>>> Dne 02.08.2021 v 9:49 as5s napsal(a):
>>>>>
>>>>> Dobry den,
>>>>>
>>>>> skuste pozriet na vTaskDelayUntil()
>>>>>
>>>>> https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/system/freertos.html#_CPPv415vTaskDelayUntilPC10TickType_tK10TickType_t
>>>>> <https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/system/freertos.html#_CPPv415vTaskDelayUntilPC10TickType_tK10TickType_t>
>>>>>
>>>>> mozno vam to bude stacit.
>>>>>
>>>>> Myre
>>>>>
>>>>> On 2. 8. 2021 9:45, Jan Půhoný wrote:
>>>>>> Dobrý den,
>>>>>>
>>>>>> mám tady další zádrhel - mějme funkci, která ovládá výstupní
>>>>>> piny, potřebuji po nastavení udžet cca 20 sekund v HIGH a pak
>>>>>> vrátit vše do low. Jedná se o otevítání a zavírání ventilu.
>>>>>> Přidal jsem tedy mezi to vTaskDelay(20000 /
>>>>>> portTICK_PERIOD_MS); , problém je v tom, že někdy se to
>>>>>> zpoždění značně zkrátí a ventil zůstane napolo otevřený. Jak
>>>>>> se toto dá ošetřit aby se vTaskDelay(20000 /
>>>>>> portTICK_PERIOD_MS); nikdy nemohlo zkrátit a bylo dodrženo.
>>>>>>
>>>>>> Volám to pak standardně přes xTaskCreatePinnedToCore() ....
>>>>>>
>>>>>> void vent1On(void * parameter){
>>>>>> for(;;){ // infinite loop
>>>>>> button1.update(1);
>>>>>> digitalWrite(ventMain1, HIGH);
>>>>>> digitalWrite(ventOn1, HIGH);
>>>>>> digitalWrite(ledPinRed, HIGH);
>>>>>> // Pause the task fo 20s
>>>>>> vTaskDelay(20000 / portTICK_PERIOD_MS);
>>>>>> digitalWrite(ventMain1, LOW);
>>>>>> digitalWrite(ventOn1, LOW);
>>>>>> digitalWrite(ledPinRed, LOW);
>>>>>> vTaskDelete(NULL); //spustí se jen jednou
>>>>>> }
>>>>>> }
>>>>>>
>>>>>>
>>>>>> xTaskCreatePinnedToCore(
>>>>>> vent1On, // Function that should be called
>>>>>> "vent1On", // Name of the task (for debugging)
>>>>>> 1000, // Stack size (bytes)
>>>>>> NULL, // Parameter to pass
>>>>>> 1, // Task priority
>>>>>> NULL, // Task handle
>>>>>> 0 // Core 1/0
>>>>>> );
>>>>>>
>>>>>> HP
>>>>>>
>>>>>>
>>>>>> so 31. 7. 2021 v 8:27 odesílatel Jan Půhoný
>>>>>> <konference na puhy.cz <mailto:konference na puhy.cz>> napsal:
>>>>>>
>>>>>> Díky, zkusil jsem jak 2000 tak 4000 ale stále stejné.
>>>>>> Resetuje se to pořád dokola s tímto:
>>>>>>
>>>>>> rst:0x1 (POWERON_RESET),boot:0x12 (SPI_FAST_FLASH_BOOT)
>>>>>> configsip: 0, SPIWP:0xee
>>>>>> clk_drv:0x00,q_drv:0x00,d_drv:0x00,cs0_drv:0x00,hd_drv:0x00,wp_drv:0x00
>>>>>> mode:DIO, clock div:2
>>>>>> load:0x3fff0018,len:4
>>>>>> load:0x3fff001c,len:1044
>>>>>> load:0x40078000,len:10124
>>>>>> load:0x40080400,len:5828
>>>>>> entry 0x400806a8
>>>>>>
>>>>>> *Ale zjistil jsem, že to dělá jen pokud to napájím z USB.
>>>>>> Když to dám přímo do svého zapojení, kde je pořádná
>>>>>> kapacita na Vin, tak to drží. Takže skutečně jen problém
>>>>>> s napájením.*
>>>>>>
>>>>>> Teď už to budu programovat jen přes OTA, takže vyřešeno.
>>>>>>
>>>>>> HP
>>>>>>
>>>>>>
>>>>>> so 31. 7. 2021 v 6:44 odesílatel Jaroslav Buchta
>>>>>> <jaroslav.buchta na hascomp.cz
>>>>>> <mailto:jaroslav.buchta na hascomp.cz>> napsal:
>>>>>>
>>>>>> kod jsem nezkoumal ale na prvni pohled mi prijde maly
>>>>>> zasobnik pro
>>>>>> vlakno, tady je to tusim v btech. dal bych 2 nebo 4K
>>>>>> na zkousku
>>>>>> _______________________________________________
>>>>>> HW-list mailing list - sponsored by www.HW.cz
>>>>>> <http://www.HW.cz>
>>>>>> Hw-list na list.hw.cz <mailto:Hw-list na list.hw.cz>
>>>>>> http://list.hw.cz/mailman/listinfo/hw-list
>>>>>> <http://list.hw.cz/mailman/listinfo/hw-list>
>>>>>>
>>>>>>
>>>>>> _______________________________________________
>>>>>> HW-list mailing list - sponsored bywww.HW.cz <http://www.HW.cz>
>>>>>> Hw-list na list.hw.cz <mailto:Hw-list na list.hw.cz>
>>>>>> http://list.hw.cz/mailman/listinfo/hw-list <http://list.hw.cz/mailman/listinfo/hw-list>
>>>>>
>>>>> _______________________________________________
>>>>> HW-list mailing list - sponsored bywww.HW.cz <http://www.HW.cz>
>>>>> Hw-list na list.hw.cz <mailto:Hw-list na list.hw.cz>
>>>>> http://list.hw.cz/mailman/listinfo/hw-list <http://list.hw.cz/mailman/listinfo/hw-list>
>>>>
>>>>
>>>>
>>>> _______________________________________________
>>>> HW-list mailing list - sponsored bywww.HW.cz <http://www.HW.cz>
>>>> Hw-list na list.hw.cz <mailto:Hw-list na list.hw.cz>
>>>> http://list.hw.cz/mailman/listinfo/hw-list <http://list.hw.cz/mailman/listinfo/hw-list>
>>>
>>> _______________________________________________
>>> HW-list mailing list - sponsored bywww.HW.cz <http://www.HW.cz>
>>> Hw-list na list.hw.cz <mailto:Hw-list na list.hw.cz>
>>> http://list.hw.cz/mailman/listinfo/hw-list <http://list.hw.cz/mailman/listinfo/hw-list>
>>
>>
>>
>> _______________________________________________
>> HW-list mailing list - sponsored bywww.HW.cz <http://www.HW.cz>
>> Hw-list na list.hw.cz <mailto:Hw-list na list.hw.cz>
>> http://list.hw.cz/mailman/listinfo/hw-list <http://list.hw.cz/mailman/listinfo/hw-list>
>
>
> _______________________________________________
> HW-list mailing list - sponsored by www.HW.cz <http://www.HW.cz>
> Hw-list na list.hw.cz <mailto:Hw-list na list.hw.cz>
> http://list.hw.cz/mailman/listinfo/hw-list
> <http://list.hw.cz/mailman/listinfo/hw-list>
>
>
> _______________________________________________
> HW-list mailing list - sponsored by www.HW.cz
> Hw-list na list.hw.cz
> http://list.hw.cz/mailman/listinfo/hw-list
------------- další část ---------------
HTML příloha byla odstraněna...
URL: <http://list.hw.cz/pipermail/hw-list/attachments/20210803/739a1500/attachment-0001.html>
Další informace o konferenci Hw-list