/* * Copyright (c) 2022 OpenLuat & AirM2M * * Permission is hereby granted, 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. */ #include "user.h" typedef struct { uint32_t LastValue; uint8_t Data[6]; uint8_t Pos; uint8_t BitPos; uint8_t BytePos; }DHT11_CtrlStruct; static int32_t DHT11_ReadBit(void *pData, void *pParam) { OPQueue_CmdStruct *Cmd = (OPQueue_CmdStruct *)pData; DHT11_CtrlStruct *DHT11 = (DHT11_CtrlStruct *)pParam; uint32_t diff; switch(DHT11->Pos) { case 0: // DBG("%u", Cmd->uParam.MaxCnt); break; case 1: // DBG("%u", Cmd->uParam.MaxCnt); DHT11->LastValue = Cmd->uParam.MaxCnt; DHT11->BitPos = 7; DHT11->BytePos = 0; DHT11->Data[DHT11->BytePos] = 0; break; default: diff = (Cmd->uParam.MaxCnt - DHT11->LastValue); DHT11->LastValue = Cmd->uParam.MaxCnt; // DBG("%u,%u,%u,%u", DHT11->BytePos, DHT11->BitPos, diff, Cmd->uParam.MaxCnt); if (diff > (100 * SYS_TIMER_1US)) { DHT11->Data[DHT11->BytePos] |= (1 << DHT11->BitPos); } if (DHT11->BitPos) { DHT11->BitPos--; } else { DHT11->BitPos = 7; DHT11->BytePos++; DHT11->Data[DHT11->BytePos] = 0; } } DHT11->Pos++; } void DHT11_TestOnce(uint8_t Pin, CBFuncEx_t CB) { uint8_t HWTimerID = HW_TIMER0; uint32_t i; OPQueue_CmdStruct OPCmd; HANDLE TaskHandle = Task_GetCurrent(); HWTimer_InitOperationQueue(HWTimerID, 100, 1, 0, CB, TaskHandle); OPCmd.Operation = OP_QUEUE_CMD_SET_GPIO_DIR_OUT; OPCmd.Arg1 = Pin; OPCmd.uArg.IOArg.Level = 0; OPCmd.uArg.IOArg.PullMode = OP_QUEUE_CMD_IO_PULL_UP; HWTimer_AddOperation(HWTimerID, &OPCmd); OPCmd.Operation = OP_QUEUE_CMD_ONE_TIME_DELAY; OPCmd.Arg1 = 0; OPCmd.uArg.Time = 20000; HWTimer_AddOperation(HWTimerID, &OPCmd); OPCmd.Operation = OP_QUEUE_CMD_CAPTURE_SET; OPCmd.Arg1 = Pin; OPCmd.uArg.ExitArg.ExtiMode = OP_QUEUE_CMD_IO_EXTI_DOWN; OPCmd.uArg.ExitArg.PullMode = OP_QUEUE_CMD_IO_PULL_UP; OPCmd.uParam.MaxCnt = 100 * SYS_TIMER_1MS; HWTimer_AddOperation(HWTimerID, &OPCmd); OPCmd.Operation = OP_QUEUE_CMD_CAPTURE; OPCmd.Arg1 = Pin; OPCmd.uArg.IOArg.Level = 0xff; for(i = 0; i < 43; i++) //40bit data+2bit start+1bit end { HWTimer_AddOperation(HWTimerID, &OPCmd); } OPCmd.Operation = OP_QUEUE_CMD_CAPTURE_END; OPCmd.Arg1 = Pin; HWTimer_AddOperation(HWTimerID, &OPCmd); HWTimer_StartOperationQueue(HWTimerID); } void DHT11_TestResult(void) { uint8_t HWTimerID = HW_TIMER0; DHT11_CtrlStruct DHT11; DHT11.Pos = 0; HWTimer_GetOperationQueueCaptureResult(HWTimerID, DHT11_ReadBit, &DHT11); DHT11.Data[5] = SumCheck(DHT11.Data, 4); if (DHT11.Data[4] == DHT11.Data[5]) { DBG("湿度 %d.%d, 温度 %d.%d", DHT11.Data[0], DHT11.Data[1], DHT11.Data[2], DHT11.Data[3]); } else { DBG("check error %x,%x", DHT11.Data[4], DHT11.Data[5]); DBG_HexPrintf(DHT11.Data, 4); } }