main.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558
  1. #include "led.h"
  2. #include "delay.h"
  3. #include "sys.h"
  4. #include "stm32f10x.h"
  5. #include "stdio.h"
  6. #include "key.h"
  7. #include "usart.h"
  8. #include "drive_dht11.h"
  9. #include "bh1750.h"
  10. #include "ds1302.h"
  11. #include "oled.h"
  12. #include "sk9822.h"
  13. #include "string.h"
  14. u8 humi=0;//湿度值
  15. u8 temp=0;//温度值
  16. u8 buffer[10]={0};
  17. u16 GzVal=0; //实际光照值
  18. char ShowBuf[16]={0};
  19. char DateShowBuf[30]={0};
  20. char SendBuff[50]={0};
  21. short TEMP = 0;
  22. u16 cj_val=0;//测距值
  23. u8 openflag=0;//手动模式开关按钮
  24. u8 mode_flag=0;
  25. #define YP1 PCout(13)
  26. #define YP2 PCout(14)
  27. #define YP3 PCout(15)
  28. u8 warning_cnt=0;
  29. u16 GetTimer=0;//采集时间计数器
  30. u8 dw_val=0;//挡位//亮度值
  31. u8 sc_val=0;//色彩
  32. int set_s1=7,set_f1=0; //设置 时间1
  33. int cnt_s1=0,cnt_f1=0,cnt_m1=0;//学习时间 时分秒
  34. #define RT_INPUT PBin(8)//人体红外感应
  35. #define YP_BUSY PAin(6)
  36. #define KEY_GPIO_PROT GPIOA
  37. #define KEY1_GPIO_PIN GPIO_Pin_0
  38. #define KEY2_GPIO_PIN GPIO_Pin_1
  39. #define KEY3_GPIO_PIN GPIO_Pin_2
  40. #define KEY4_GPIO_PIN GPIO_Pin_3
  41. #define KEY5_GPIO_PIN GPIO_Pin_4
  42. #define KEY1_IN GPIO_ReadInputDataBit(KEY_GPIO_PROT,KEY1_GPIO_PIN)
  43. #define KEY2_IN GPIO_ReadInputDataBit(KEY_GPIO_PROT,KEY2_GPIO_PIN)
  44. #define KEY3_IN GPIO_ReadInputDataBit(KEY_GPIO_PROT,KEY3_GPIO_PIN)
  45. #define KEY4_IN GPIO_ReadInputDataBit(KEY_GPIO_PROT,KEY4_GPIO_PIN)
  46. #define KEY5_IN GPIO_ReadInputDataBit(KEY_GPIO_PROT,KEY5_GPIO_PIN)
  47. void key_gpio_init(void)
  48. {
  49. GPIO_InitTypeDef GPIO_InitStructure;
  50. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC, ENABLE);
  51. GPIO_InitStructure.GPIO_Pin = KEY1_GPIO_PIN|KEY2_GPIO_PIN|KEY3_GPIO_PIN|KEY4_GPIO_PIN|KEY5_GPIO_PIN|GPIO_Pin_6;
  52. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
  53. GPIO_Init(KEY_GPIO_PROT, &GPIO_InitStructure);
  54. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
  55. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  56. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  57. GPIO_Init(GPIOA, &GPIO_InitStructure);
  58. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
  59. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  60. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  61. GPIO_Init(GPIOA, &GPIO_InitStructure);
  62. }
  63. #define Trig PBout(10) // 超声波测距
  64. #define CW_ECHO1 PBin(11)
  65. u16 Distance=0;
  66. //超声波端口初始化
  67. void Wave_IO_Init(void)
  68. {
  69. GPIO_InitTypeDef GPIO_InitStructure;
  70. // NVIC_InitTypeDef NVIC_InitStructure;
  71. TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
  72. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2 , ENABLE);
  73. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB , ENABLE);
  74. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //PB0接TRIG
  75. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //设为推挽输出模式
  76. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  77. GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化外设GPIO
  78. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11; //接ECH0
  79. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //设为输入
  80. GPIO_Init(GPIOB, &GPIO_InitStructure);
  81. TIM_DeInit(TIM2);
  82. TIM_TimeBaseStructure.TIM_Period=65535; /* 自动重装载寄存器周期的值(计数值) */
  83. /* 累计 TIM_Period个频率后产生一个更新或者中断 */
  84. TIM_TimeBaseStructure.TIM_Prescaler= (360 - 1); /* 时钟预分频数 72M/360 */
  85. TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1; /* 采样分频 */
  86. TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up; /* 向上计数模式 */
  87. TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
  88. TIM_ClearFlag(TIM2, TIM_FLAG_Update); /* 清除溢出中断标志 */
  89. // NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);
  90. // NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
  91. // NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  92. // NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;
  93. // NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  94. // NVIC_Init(&NVIC_InitStructure);
  95. //
  96. //TIM2_NVIC_Configuration();//定时器初始化
  97. //TIM2_Configuration();
  98. }
  99. //void TIM2_IRQHandler()
  100. //{
  101. // static u8 st;
  102. // st=TIM_GetFlagStatus(TIM2, TIM_IT_Update);
  103. // if(st!=0)
  104. // {
  105. // TIM_ClearFlag(TIM2, TIM_IT_Update);
  106. // }
  107. //}
  108. //测距
  109. //开始测距,发送一个>10us的脉冲,然后测量返回的高电平时间
  110. u16 Wave_Start(void)
  111. {
  112. u32 Distance;
  113. u32 timeout=0;
  114. Trig=1;
  115. delay_us(20); //输入一个10us的高电平
  116. Trig=0;
  117. TIM_SetCounter(TIM2,0);
  118. timeout=0;
  119. while(!CW_ECHO1) //等待高电平
  120. {
  121. if(++timeout>60000)break;
  122. delay_us(1);
  123. }
  124. TIM_Cmd(TIM2, ENABLE); //开启时钟
  125. timeout=0;
  126. while(CW_ECHO1) //等待低电平
  127. {
  128. if(++timeout>600000)break;
  129. delay_us(1);
  130. }
  131. TIM_Cmd(TIM2, DISABLE); //定时器2失能
  132. Distance=TIM_GetCounter(TIM2)*5*34/2000; //计算距离 厘米cm
  133. TIM_SetCounter(TIM2,0);
  134. if(Distance>200)Distance=200;
  135. return Distance;
  136. }
  137. void TIMER3_Int_Init(u16 arr,u16 psc)
  138. {
  139. TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStrue;
  140. NVIC_InitTypeDef NVIC_InitStructure;
  141. TIM_TimeBaseInitStrue.TIM_Period=arr;
  142. TIM_TimeBaseInitStrue.TIM_Prescaler=psc;
  143. TIM_TimeBaseInitStrue.TIM_CounterMode=TIM_CounterMode_Up;
  144. TIM_TimeBaseInitStrue.TIM_ClockDivision=TIM_CounterMode_Up;
  145. RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
  146. TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStrue);
  147. TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE);//使能TIM3更新中断
  148. //中断优先级NVIC设置
  149. NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中断
  150. NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占优先级0级
  151. NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //从优先级3级
  152. NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能
  153. NVIC_Init(&NVIC_InitStructure); //初始化NVIC寄存器
  154. TIM_Cmd(TIM3, ENABLE); //使能TIMx
  155. }
  156. char clear_flag=0;
  157. int key_Task(void)
  158. {
  159. static char key_flag=0;
  160. if(KEY1_IN==0|KEY2_IN==0|KEY3_IN==0|KEY4_IN==0|KEY5_IN==0)
  161. {
  162. if(key_flag==0)
  163. {
  164. key_flag=1;
  165. if(KEY1_IN==0)
  166. {
  167. mode_flag^=1;
  168. }
  169. else if(KEY2_IN==0)
  170. {
  171. if(mode_flag==0)
  172. {
  173. if(++dw_val>2)dw_val=0;
  174. }else
  175. {
  176. cnt_s1++;
  177. }
  178. }
  179. else if(KEY3_IN==0)
  180. {
  181. if(mode_flag==0)
  182. {
  183. if(++sc_val>2)sc_val=0;
  184. }else
  185. {
  186. if(cnt_s1>0)cnt_s1--;
  187. }
  188. }
  189. else if(KEY4_IN==0)
  190. {
  191. if(mode_flag==0)
  192. {
  193. openflag^=1;//开关按钮
  194. }else
  195. {
  196. }
  197. }
  198. else if(KEY5_IN==0)
  199. {
  200. clear_flag^=1;
  201. }
  202. }
  203. }else
  204. {
  205. key_flag=0;
  206. }
  207. return 20;
  208. }
  209. u16 DataUpTime=0;//时间显示计数器
  210. u8 rt_flag=0;//红外人体标志位
  211. u16 rt_cnt=0;
  212. u8 seccnt=0;
  213. void TIM3_IRQHandler(void)//中断服务程序
  214. {
  215. if(TIM_GetITStatus(TIM3,TIM_IT_Update)==SET)//SET为1
  216. {
  217. GetTimer++;//采集计数器
  218. TIM_ClearITPendingBit(TIM3,TIM_IT_Update); //清除TIM3的更新中断标志
  219. key_Task();
  220. DataUpTime++;
  221. if(RT_INPUT==1) //检测人体感应
  222. {
  223. rt_flag=1;
  224. rt_cnt=0;
  225. }else
  226. {
  227. if(rt_cnt++>50*15)//15s没人了
  228. {
  229. rt_cnt=0;rt_flag=0;
  230. }
  231. }
  232. if(rt_flag) //有人 开始计时计算学习时间
  233. {
  234. if(++seccnt>=50)
  235. {
  236. seccnt=0;
  237. if(++cnt_m1>59)
  238. {
  239. cnt_m1=0;
  240. if(++cnt_f1>59)
  241. {
  242. cnt_f1=0;
  243. if(++cnt_s1>23)
  244. {
  245. cnt_s1=0;
  246. }
  247. }
  248. }
  249. }
  250. }
  251. }
  252. }
  253. u8 GetFlag=0;
  254. uint8_t sendBuff[16]={0};
  255. void ShowTo(char* Pdat,char len)//字符串中的空格补 0
  256. {
  257. char ixd=0;
  258. for(ixd=0;ixd<len;ixd++)
  259. {
  260. if(Pdat[ixd]==' ')
  261. {
  262. Pdat[ixd]='0';
  263. }
  264. }
  265. }
  266. #define IO_TX_GPIO_Port PAout(7)
  267. void uartX_sendByte(uint8_t val)
  268. {
  269. int i = 0;
  270. IO_TX_GPIO_Port=0;
  271. delay_us(104);
  272. for (i = 0; i < 8; i++)
  273. {
  274. if (val & 0x01)
  275. {
  276. IO_TX_GPIO_Port=1;
  277. } else {
  278. IO_TX_GPIO_Port=0;
  279. }
  280. delay_us(104);
  281. val >>= 1;
  282. }
  283. IO_TX_GPIO_Port=1;
  284. delay_us(104);
  285. }
  286. void sendData(char *p,unsigned char n)
  287. {
  288. if( p == 0) return ;
  289. if(n > 0)
  290. {
  291. while(n --)
  292. {
  293. uartX_sendByte(*p++);
  294. }
  295. }
  296. }
  297. u8 test=0;
  298. unsigned char sendbuff1[8]={0x7E, 0x04 ,0x03 ,0x00, 0x01, 0xEF};
  299. unsigned char sendbuff2[8]={0x7E, 0x04 ,0x03 ,0x00, 0x02, 0xEF};
  300. unsigned char sendbuff3[8]={0x7E, 0x04 ,0x03 ,0x00, 0x03, 0xEF};
  301. int main(void)
  302. {
  303. NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级
  304. delay_init(); //延时函数初始化
  305. LED_Init(); //初始化与连接的硬件接口
  306. YP1=1;YP2=1;YP3=1;
  307. uart_init(9600);
  308. delay_ms(100);
  309. key_gpio_init();
  310. TIMER3_Int_Init(20000,71);//定时器初始化 //10ms进一次中断
  311. BH1750_Init();
  312. OLED_Init();//液晶初始化
  313. OLED_Clear();//液晶清屏
  314. delay_ms(10);//上电延时一会
  315. Ds1302_Gpio_Init();
  316. SK9822_Intit();
  317. Wave_IO_Init();
  318. SK9822_Write_Data(0xff,0,0,0);
  319. // Ds1302_Write_Time_All(1);
  320. delay_ms(10);
  321. while(1)
  322. {
  323. if(GetTimer>50)//1S
  324. {
  325. GetTimer=0;
  326. if(openflag)
  327. {
  328. switch(dw_val)
  329. {
  330. case 0:
  331. switch(sc_val)
  332. {
  333. case 0:
  334. SK9822_Write_Data(0xff,0xff,0xff,0xff);
  335. break;
  336. case 1:
  337. SK9822_Write_Data(0xff,255,255,0);
  338. break ;
  339. case 2:
  340. SK9822_Write_Data(0xff,60,255,60);
  341. break;
  342. }
  343. break;
  344. case 1:
  345. switch(sc_val)
  346. {
  347. case 0:
  348. SK9822_Write_Data(0xff,125,125,125);
  349. break;
  350. case 1:
  351. SK9822_Write_Data(0xff,125,125,0);
  352. break ;
  353. case 2:
  354. SK9822_Write_Data(0xff,30,125,30);
  355. break;
  356. }
  357. break;
  358. case 2:
  359. switch(sc_val)
  360. {
  361. case 0:
  362. SK9822_Write_Data(0xff,30,30,30);
  363. break;
  364. case 1:
  365. SK9822_Write_Data(0xff,40,40,0);
  366. break ;
  367. case 2:
  368. SK9822_Write_Data(0xff,10,50,10);
  369. break;
  370. }
  371. break;
  372. }
  373. }else{
  374. SK9822_Write_Data(0xff,0,0,0);//熄灭
  375. }
  376. if(mode_flag)//自动模式下控制
  377. {
  378. OLED_ShowString(90,6,"A",16);
  379. }else{ //手动模式下控制
  380. OLED_ShowString(90,6,"M",16);
  381. }
  382. if(dht11_read_data(buffer) == 0)//读取温湿度
  383. {
  384. humi = buffer[0] + buffer[1] / 10;//取湿度值
  385. temp = buffer[2] + buffer[3] / 10;//取温度值
  386. }
  387. GzVal = Dispose();//读取光照
  388. cj_val=Wave_Start();
  389. if(cj_val>15)
  390. {
  391. OLED_ShowText(0,6,"坐姿:异常",0);
  392. warning_cnt|=0x01;
  393. }else
  394. {
  395. warning_cnt&=~0x01;
  396. OLED_ShowText(0,6,"坐姿:正常",0);
  397. }
  398. if(rt_flag) //有人
  399. {
  400. OLED_ShowString(103,6,"1",16);
  401. }
  402. else //没人
  403. {
  404. OLED_ShowString(103,6,"0",16);
  405. }
  406. printf("drv_X%dH%dD%dZ%d\r\n",temp,humi,GzVal,cj_val);//发送到上位机app
  407. if(mode_flag)//自动模式
  408. {
  409. if(rt_flag) //有人
  410. {
  411. openflag=1;
  412. if(GzVal<50)
  413. {
  414. dw_val=0;//开启最大亮度
  415. }
  416. else if(GzVal>50&&GzVal<500)
  417. {
  418. dw_val=1;
  419. }
  420. else if(GzVal>500)
  421. {
  422. dw_val=2;
  423. }
  424. }else
  425. {
  426. openflag=0;//关闭
  427. }
  428. }else{ //手动模式
  429. }
  430. }
  431. if(DataUpTime>20) //400ms刷新一次数据
  432. {
  433. DataUpTime=0;//
  434. Ds1302_Readtime();
  435. sprintf((char*)DateShowBuf,"时间:%2d:%2d:%2d", (int)ds1302Data.hour, (int)ds1302Data.min, (int)ds1302Data.sec);
  436. ShowTo(DateShowBuf,strlen(DateShowBuf));
  437. OLED_ShowText(0,0,DateShowBuf,0);
  438. sprintf((char*)DateShowBuf,"闹钟:%2d:%2d:00", (int)set_s1, (int)set_f1);
  439. ShowTo(DateShowBuf,strlen(DateShowBuf));
  440. OLED_ShowText(0,2,DateShowBuf,0);
  441. sprintf((char*)DateShowBuf,"学习:%2d:%2d:%2d", (int)cnt_s1, (int)cnt_f1,cnt_m1);
  442. ShowTo(DateShowBuf,strlen(DateShowBuf));
  443. OLED_ShowText(0,4,DateShowBuf,0);
  444. if((ds1302Data.hour==set_s1)&&(ds1302Data.min==set_f1)) //到达闹钟时间
  445. {
  446. warning_cnt|=0x02;
  447. }
  448. else
  449. {
  450. warning_cnt&=~0x02;
  451. }
  452. if(cnt_s1>=2)
  453. {
  454. warning_cnt|=0x04;
  455. }else{
  456. warning_cnt&=~0x04;
  457. }
  458. if(clear_flag==0)//取消提醒标志位
  459. {
  460. if((warning_cnt&0x01)==0x01) //坐姿异常
  461. {
  462. if(YP_BUSY==0)
  463. {
  464. sendData((char*)sendbuff1,6);
  465. }
  466. }
  467. else if((warning_cnt&0x02)==0x02)//闹钟
  468. {
  469. if(YP_BUSY==0)
  470. {
  471. sendData((char*)sendbuff2,6);
  472. }
  473. }
  474. else if((warning_cnt&0x04)==0x04)//学习时间过长
  475. {
  476. if(YP_BUSY==0)
  477. sendData((char*)sendbuff3,6);
  478. }
  479. }
  480. }
  481. }
  482. }