日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > windows >内容正文

windows

一步步学习操作系统(1)——参照ucos,在STM32上实现一个简单的多任务(“啰里啰嗦版”)...

發布時間:2024/4/15 windows 79 豆豆
生活随笔 收集整理的這篇文章主要介紹了 一步步学习操作系统(1)——参照ucos,在STM32上实现一个简单的多任务(“啰里啰嗦版”)... 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

該篇為“啰里啰嗦版”,另有相應的“精簡版”供參考

?

“不到長城非好漢;不做OS,枉為程序員”

OS之于程序員,如同梵蒂岡之于天主教徒,那永遠都是塊神圣的領土。若今生不能親歷之,實乃憾事!

但是,圣域不是想進就能進的呀……

OS融合了大量的計算機的基礎知識,各個知識領域之間交織緊密,初來乍到者一不小心就會繞出個死結。

我的方法是:死結就死結,不管三七二十一,直接剪斷,先走下去再說,回頭我們再把這個剪斷的死結再接上。

?

我們知道,“多任務”一般在介紹OS的書籍中,是屬于中間或者靠后的部分,又或者分散在各個章節中。而我決定上手就說它。

?

一、整體縱覽:

1、硬件:

STM32F103RC

2、IDE:

MDK5

3、文件架構:

(1)標準文件:

startup_stm32f10x_hd.s:STM32官方啟動文件(注意:這是針對stm32硬件配置的文件,型號要是不同,你的可能和我不一樣哦

(2)自編文件——這才是我們的重頭戲哦(共5個文件:1x".asm"+2x".c"+2x".h"):

main.c:主函數和任務定義;

os_cpu_a.asm:中斷與任務切換;

myos.c:硬件初始化與任務切換時的堆棧保存;

include.h和myos.h:兩個頭文件。

?

二、逐文解析:

1、main.c

順著main函數這條主線,我們看到最終的OS其實就是執行了7行代碼,共5個數:

1、OSInit(); 2、OSTaskCreate(Task1, (void*)0, (OS_STK*)&Task1Stk[TASK_STACK_SIZE-1]); 3、OSTaskCreate(Task2, (void*)0, (OS_STK*)&Task2Stk[TASK_STACK_SIZE-1]); 4、OSTaskCreate(Task3, (void*)0, (OS_STK*)&Task3Stk[TASK_STACK_SIZE-1]); 5、SysTickInit(5); 6、LedInit(); 7、OSStart();

簡單說說main函數功能:
當OSStart()執行之后,Task1、Task2、Task3輪流執行,即Task1()、Task2()、Task3()三個函數輪流執行:
Task1()->Task2()->Task3()->Task1()->Task2()->Task3()->Task1()->......
每個任務(或函數)執行相等的時間片,并有SysTick中斷來觸發PendSV中斷,從而實現任務切換。
OSStart()執行之后,永不返回。

各個函數基本做了些什么,代碼后面都附加了注解。
其中需要注意的地方是:OSStart()。
這個函數一旦執行了就不會返還,有點像死循環(但不是死循環哦,后來會明白的)。
仔細想想后,確實也應當如此,如果main函數return掉了的話,程序也就結束啦!
“main函數結束”就意味著CPU現在只會喝喝茶、看看報了,什么搶劫、著火它都裝作沒看見。

main函數就這么短,那么上面七個函數的實現在哪里呢?
這時你肯定想到"#include"了吧!
“太好了!#include只有一行!”
那我們去include.h那里看看吧。
1 #include "include.h" 2 extern OS_TCB OSTCBTbl[OS_MAX_TASKS]; // (OS Task Control Block Table) 3 extern OS_STK TASK_IDLE_STK[TASK_STACK_SIZE]; //("TaskIdle" Stack) 4 extern OS_TCB *OSTCBCur; // Pointer to the current running task(OS Task Control Block Current) 5 extern OS_TCB *OSTCBNext; // Pointer to the next running task(OS Task Control Block Next) 6 extern INT8U OSTaskNext; // Index of the next task 7 extern INT32U TaskTickLeft; // Refer to the time ticks left for the current task 8 extern INT32U TimeMS; // For system time record 9 extern INT32U TaskTimeSlice; // For system time record 10 11 OS_STK Task1Stk[TASK_STACK_SIZE]; // initialize stack for task1 12 OS_STK Task2Stk[TASK_STACK_SIZE]; // initialize stack for task2 13 OS_STK Task3Stk[TASK_STACK_SIZE]; // initialize stack for task3 14 15 void Task1(void *p_arg); // flip the led1 every 0.5s 16 void Task2(void *p_arg); // flip the led2 every 1.0s 17 void Task3(void *p_arg); // do nothing 18 19 int main(void) 20 { 21 22 23 OSInit(); // OS initialization 24 OSTaskCreate(Task1, (void*)0, (OS_STK*)&Task1Stk[TASK_STACK_SIZE-1]); // create task 1 25 OSTaskCreate(Task2, (void*)0, (OS_STK*)&Task2Stk[TASK_STACK_SIZE-1]); // create task 2 26 OSTaskCreate(Task3, (void*)0, (OS_STK*)&Task3Stk[TASK_STACK_SIZE-1]); // create task 3 27 SysTickInit(5); // configure the SysTick as 5ms 28 LedInit(); // leds initialization 29 OSStart(); // start os! 30 31 return 0; // never come here 32 } 33 34 void Task1(void *p_arg) 35 { 36 while(1) { 37 delayMs(100); // delay 100 * 5ms = 0.5s 38 LED1TURN(); // flip the switch of led1 39 } 40 } 41 void Task2(void *p_arg) 42 { 43 while(1) { 44 delayMs(200); // delay 200 * 5ms = 1.0s 45 LED2TURN(); // flip the switch of led2 46 } 47 } 48 49 void Task3(void *p_arg) 50 { 51 while(1) { 52 } 53 }

小白兔筆記:

(1)“啥是extern變量啊?”

"快去復習復習c語言教程吧。"

(2)"OS_TCB、OS_MAX_TASKS什么的都是些啥?“

"myos.h"里都有它們的定義。

(3)”'OSTCBCur'都是些啥怪名字?"

“針對詞義復雜的變量,注意看定義那行注釋,后面的括號會有對變量的簡短說明,如:

extern OS_TCB *OSTCBCur; // Pointer to the current running task(OS Task Control Block Current)

?

?

?2、include.h

“我去!這不是欺騙我感情嗎?main函數倒是1個#include,怎么到了這里卻又來了三個!”

等等!先別灰心嘛,容我慢慢道來。

我們知道,main里的

#include "include.h"

其實就等于:

#include <stdlib.h>

#include "myos.h"

#include "stm32f10x.h"

但是我們為什么要那么費勁再弄一個"inlcude.h"文件呢?

假設我們想給這個OS再加個內存管理的功能,于是要添加幾個".c"文件,而這些文件也要包含上面那幾個"#include",那么我們不是要再把那幾個“#include”都寫一遍嗎??

不過,有了這個include.h之后,這些".c"文件只要

#include "include.h"就可以了。

多加了1個"include.h",但卻清爽了n個”xxxxx.c"文件。

當然,這只是好處之一,其他的就不多說了,畢竟我們的主題是OS嘛。

1 #ifndef INCLUDE_H 2 #define INCLUDE_H 3 4 #include <stdlib.h> 5 #include "myos.h" 6 #include "stm32f10x.h" //stm32官方頭文件 7 8 #endif

小白兔筆記:

(1)“#ifndef INCLUDE_H之類的東西是什么意思?”

這是為了防止多重包含頭文件。

既然你問了這個問題,估計你也聽不懂啥是”多重包含頭文件“。

這是和編譯器有關的約定,簡單點說,不這么寫,編譯器”可能“——我說”可能“——找你茬。

(2)”stm32f10x.h“是官方根據stm32的各種硬件配置編寫的頭文件,可要找準著你的你自己的硬件配置使用哦!

?

3、myos.h

1 #ifndef MYOS_H 2 #define MYOS_H 3 #include "stm32f10x.h" 4 5 /**********CPU DEPENDENT************/ 6 #define TASK_TIME_SLICE 5 // 5ms for every task to run every time 7 8 typedef unsigned char INT8U; // Unsigned 8 bit quantity 9 typedef unsigned short INT16U; // Unsigned 16 bit quantity 10 typedef unsigned int INT32U; // Unsigned 32 bit quantity 11 12 typedef unsigned int OS_STK; // Each stack entry is 32-bit wide(OS Stack) 13 14 // assembling functions 15 void OS_ENTER_CRITICAL(void); // Enter Critical area, that is to disable interruptions 16 void OS_EXIT_CRITICAL(void); // Exit Critical area, that is to enable interruptions 17 void OSCtxSw(void); // Task Switching Function(OS Context Switch) 18 void OSStart(void); 19 20 OS_STK* OSTaskStkInit(void (*task)(void *p_arg), // task function 21 void *p_arg, // (pointer of arguments) 22 OS_STK *p_tos); // (pointer to the top of stack) 23 /**********CPU INDEPENDENT************/ 24 25 #define OS_MAX_TASKS 16 26 27 #define TASK_STATE_CREATING 0 28 #define TASK_STATE_RUNNING 1 29 #define TASK_STATE_PAUSING 2 30 31 #define TASK_STACK_SIZE 64 32 33 #define LED1TURN() (GPIOA->ODR ^= 1<<8) // reverse the voltage of LED1 !!!HARDWARE RELATED 34 #define LED2TURN() (GPIOD->ODR ^= 1<<2) // reverse the voltage of LED2 !!!HARDWARE RELATED 35 36 37 typedef struct os_tcb { 38 OS_STK *OSTCBStkPtr; // (OS Task Control Block Stack Pointer) 39 INT8U OSTCBStat; // (OS Task Control Block Status) 40 } OS_TCB; // (OS Task Control Block) 41 42 void OSInit(void); // (OS Initialization) 43 void LedInit(void); 44 45 void OS_TaskIdle(void *p_arg); 46 void OSInitTaskIdle(void); // (OS Initialization of "TaskIdle") 47 void OSTaskCreate(void (*task)(void *p_arg), // task function 48      void *p_arg,        // (pointer of arguments) 49 OS_STK *p_tos);         // (pointer to the top of stack) 50 void OSTCBSet(OS_TCB *p_tcb, OS_STK *p_tos, INT8U task_state); 51 52 53 void SysTickInit(INT8U Nms); // (System Tick Initialization) 54 void SysTick_Handler(void);            // The interrupt function 55 56 INT32U GetTime(void); 57 void delayMs(volatile INT32U ms);         // The argument can't be too large 58 59 #endif

這是最后一個頭文件了,也是一個簡單易懂的文件。同時也是最后的平原,過了這個平原,我們可就要翻雪山啦!

(1)簡單說說2個函數:

void OS_ENTER_CRITICAL(void);

void OS_EXIT_CRITICAL(void);


有一種東西叫“臨界區”(CRITICAL),這些所謂“臨界區”指的是一些變量所在的內存,可以直接理解成“就是些特殊變量”。
要訪問這些變量必須得關掉“中斷”,訪問結束后再開啟“中斷”,開關“中斷”就是這兩個函數的任務了。
猜猜看哪個是開“中斷”,哪個是關“中斷”呢?

(2)系統時鐘中斷void SysTick_Handler(void)的由來:

來自官方啟動文件startup_stm32f10x_hd.s。

?

小白兔筆記:

小白兔表示“感覺不會再愛了……”

?

4、os_cpu_a.asm和myos.c:

上文說過,這兩個文件關系曖昧,扯開來只講其中一個很沒味道,這里先把它們都貼出來:

os_cpu_a.asm(“;”后的注釋是對應"C"語言的解釋):

1 IMPORT OSTCBCur 2 IMPORT OSTCBNext 3 4 EXPORT OS_ENTER_CRITICAL 5 EXPORT OS_EXIT_CRITICAL 6 EXPORT OSStart 7 EXPORT PendSV_Handler 8 EXPORT OSCtxSw 9 10 NVIC_INT_CTRL EQU 0xE000ED04 ; Address of NVIC Interruptions Control Register 11 NVIC_PENDSVSET EQU 0x10000000 ; Enable PendSV 12 NVIC_SYSPRI14 EQU 0xE000ED22 ; System priority register (priority 14). 13 NVIC_PENDSV_PRI EQU 0xFF ; PendSV priority value (lowest). 14 15 PRESERVE8 ; align 8 16 17 AREA |.text|, CODE, READONLY 18 THUMB 19 20 ;/******************OS_ENTER_CRITICAL************/ 21 OS_ENTER_CRITICAL 22 CPSID I ; Enable interruptions(Change Processor States: Interrupts Disable) 23 BX LR ; Return 24 25 ;/******************OS_EXIT_CRITICAL************/ 26 OS_EXIT_CRITICAL 27 CPSIE I ; Disable interruptions 28 BX LR ; Return 29 30 ;/******************OSStart************/ 31 OSStart 32 ; disable interruptions 33 CPSID I ; OS_ENTER_CRITICAL(); 34 ; initialize PendSV 35 ; Set the PendSV exception priority 36 LDR R0, =NVIC_SYSPRI14 ; R0 = NVIC_SYSPRI14; 37 LDR R1, =NVIC_PENDSV_PRI ; R1 = NVIC_PENDSV_PRI; 38 STRB R1, [R0] ; *R0 = R1; 39 40 ; initialize PSP as 0 41 ; MOV R4, #0 42 LDR R4, =0x0 ; R4 = 0; 43 MSR PSP, R4 ; PSP = R4; 44 45 ; trigger PendSV 46 LDR R4, =NVIC_INT_CTRL ; R4 = NVIC_INT_CTRL; 47 LDR R5, =NVIC_PENDSVSET ; R5 = NVIC_PENDSVSET; 48 STR R5, [R4] ; *R4 = R5; 49 50 ; enable interruptions 51 CPSIE I ; OS_EXIT_CRITICAL(); 52 53 ; should never get here 54 ; a endless loop 55 OSStartHang 56 B OSStartHang 57 58 ;/******************PendSV_Handler************/ 59 PendSV_Handler 60 CPSID I ; OS_ENTER_CRITICAL(); 61 ; judge if PSP is 0 which means the task is first invoked 62 MRS R0, PSP ; R0 = PSP; 63 CBZ R0, PendSV_Handler_NoSave ; if(R0 == 0) goto PendSV_Handler_NoSave; 64 65 ; R12, R3, R2, R1 66 SUB R0, R0, #0x20 ; R0 = R0 - 0x20; 67 68 ; store R4 69 STR R4 , [R0] ; *R0 = R4; 70 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 71 ; store R5 72 STR R5 , [R0] ; *R0 = R5; 73 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 74 ; store R6 75 STR R6 , [R0] ; *R0 = R6; 76 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 77 ; store R7 78 STR R7 , [R0] ; *R0 = R7; 79 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 80 ; store R8 81 STR R8 , [R0] ; *R0 = R8; 82 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 83 ; store R9 84 STR R9, [R0] ; *R0 = R4; 85 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 86 ; store R10 87 STR R10, [R0] ; *R0 = R10; 88 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 89 ; store R11 90 STR R11, [R0] ; *R0 = R11; 91 ADD R0, R0, #0x4 ; R0 = R0 + 0x4; 92 93 SUB R0, R0, #0x20 ; R0 = R0 - 0x20; 94 95 ; easy method 96 ;SUB R0, R0, #0x20 97 ;STM R0, {R4-R11} 98 99 LDR R1, =OSTCBCur ; R1 = OSTCBCur; 100 LDR R1, [R1] ; R1 = *R1;(R1 = OSTCBCur->OSTCBStkPtr) 101 STR R0, [R1] ; *R1 = R0;(*(OSTCBCur->OSTCBStkPrt) = R0) 102 103 PendSV_Handler_NoSave 104 LDR R0, =OSTCBCur ; R0 = OSTCBCur; 105 LDR R1, =OSTCBNext ; R1 = OSTCBNext; 106 LDR R2, [R1] ; R2 = OSTCBNext->OSTCBStkPtr; 107 STR R2, [R0] ; *R0 = R2;(OSTCBCur->OSTCBStkPtr = OSTCBNext->OSTCBStkPtr) 108 109 LDR R0, [R2] ; R0 = *R2;(R0 = OSTCBNext->OSTCBStkPtr) 110 ; LDM R0, {R4-R11} 111 ; load R4 112 LDR R4, [R0] ; R4 = *R0;(R4 = *(OSTCBNext->OSTCBStkPtr)) 113 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 114 ; load R5 115 LDR R5, [R0] ; R5 = *R0;(R5 = *(OSTCBNext->OSTCBStkPtr)) 116 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 117 ; load R6 118 LDR R6, [R0] ; R6 = *R0;(R6 = *(OSTCBNext->OSTCBStkPtr)) 119 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 120 ; load R7 121 LDR R7 , [R0] ; R7 = *R0;(R7 = *(OSTCBNext->OSTCBStkPtr)) 122 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 123 ; load R8 124 LDR R8 , [R0] ; R8 = *R0;(R8 = *(OSTCBNext->OSTCBStkPtr)) 125 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 126 ; load R9 127 LDR R9 , [R0] ; R9 = *R0;(R9 = *(OSTCBNext->OSTCBStkPtr)) 128 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 129 ; load R10 130 LDR R10 , [R0] ; R10 = *R0;(R10 = *(OSTCBNext->OSTCBStkPtr)) 131 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 132 ; load R11 133 LDR R11 , [R0] ; R11 = *R0;(R11 = *(OSTCBNext->OSTCBStkPtr)) 134 ADD R0, R0, #0x4 ; R0 = R0 + 0x4;(OSTCBNext->OSTCBStkPtr++) 135 136 MSR PSP, R0 ; PSP = R0;(PSP = OSTCBNext->OSTCBStkPtr) 137 ; P42 138 ; P139 (key word: EXC_RETURN) 139 ; use PSP 140 ORR LR, LR, #0x04 ; LR = LR | 0x04; 141 CPSIE I ; OS_EXIT_CRITICAL(); 142 BX LR ; return; 143 144 OSCtxSw ;OS context switch 145 PUSH {R4, R5} 146 LDR R4, =NVIC_INT_CTRL ; R4 = NVIC_INT_CTRL 147 LDR R5, =NVIC_PENDSVSET ; R5 = NVIC_PENDSVSET 148 STR R5, [R4] ; *R4 = R5 149 POP {R4, R5} 150 BX LR ; return; 151 152 align 4 153 end

myos.c:

1 #include "myos.h" 2 #include "stm32f10x.h" 3 4 5 OS_TCB OSTCBTbl[OS_MAX_TASKS]; // (OS Task Control Block Table) 6 OS_STK TASK_IDLE_STK[TASK_STACK_SIZE]; //("TaskIdle" Stack) 7 OS_TCB *OSTCBCur; // Pointer to the current running task(OS Task Control Block Current) 8 OS_TCB *OSTCBNext; // Pointer to the next running task(OS Task Control Block Next) 9 INT8U OSTaskNext; // Index of the next task 10 INT32U TaskTickLeft; // Refer to the time ticks left for the current task 11 INT32U TimeMS; 12 INT32U TaskTimeSlice; 13 char * Systick_priority = (char *)0xe000ed23; 14 // Initialize the stack of a task, it is of much relationship with the specific CPU 15 OS_STK* OSTaskStkInit(void (*task)(void *p_arg), 16 void *p_arg, 17 OS_STK *p_tos) 18 { 19 OS_STK *stk; 20 stk = p_tos; 21 22 *(stk) = (INT32U)0x01000000L; // xPSR 23 *(--stk) = (INT32U)task; // Entry Point 24 25 // Don't be serious with the value below. They are of random 26 *(--stk) = (INT32U)0xFFFFFFFEL; // R14 (LR) 27 *(--stk) = (INT32U)0x12121212L; // R12 28 *(--stk) = (INT32U)0x03030303L; // R3 29 *(--stk) = (INT32U)0x02020202L; // R2 30 *(--stk) = (INT32U)0x01010101L; // R1 31 32 // pointer of the argument 33 *(--stk) = (INT32U)p_arg; // R0 34 35 // Don't be serious with the value below. They are of random 36 *(--stk) = (INT32U)0x11111111L; // R11 37 *(--stk) = (INT32U)0x10101010L; // R10 38 *(--stk) = (INT32U)0x09090909L; // R9 39 *(--stk) = (INT32U)0x08080808L; // R8 40 *(--stk) = (INT32U)0x07070707L; // R7 41 *(--stk) = (INT32U)0x06060606L; // R6 42 *(--stk) = (INT32U)0x05050505L; // R5 43 *(--stk) = (INT32U)0x04040404L; // R4 44 return stk; 45 } 46 47 // Only to initialize the Task Control Block Table 48 void OSInit(void) 49 { 50 INT8U i; 51 OS_ENTER_CRITICAL(); 52 for(i = 0; i < OS_MAX_TASKS; i++) { 53 OSTCBTbl[i].OSTCBStkPtr = (OS_STK*)0; 54 OSTCBTbl[i].OSTCBStat = TASK_STATE_CREATING; 55 } 56 OSInitTaskIdle(); 57 OSTCBCur = &OSTCBTbl[0]; 58 OSTCBNext = &OSTCBTbl[0]; 59 OS_EXIT_CRITICAL(); 60 } 61 62 void OSInitTaskIdle(void) 63 { 64 OS_ENTER_CRITICAL(); 65 OSTCBTbl[0].OSTCBStkPtr = OSTaskStkInit(OS_TaskIdle, (void *)0, (OS_STK*)&TASK_IDLE_STK[TASK_STACK_SIZE - 1]); 66 OSTCBTbl[0].OSTCBStat = TASK_STATE_RUNNING; 67 OS_EXIT_CRITICAL(); 68 } 69 70 void OSTaskCreate(void (*task)(void *p_arg), 71 void *p_arg, 72 OS_STK *p_tos) 73 { 74 OS_STK * tmp; 75 INT8U i = 1; 76 OS_ENTER_CRITICAL(); 77 while(OSTCBTbl[i].OSTCBStkPtr != (OS_STK*)0) { 78 i++; 79 } 80 tmp = OSTaskStkInit(task, p_arg, p_tos); 81 OSTCBSet(&OSTCBTbl[i], tmp, TASK_STATE_CREATING); 82 OS_EXIT_CRITICAL(); 83 } 84 85 void OSTCBSet(OS_TCB *p_tcb, OS_STK *p_tos, INT8U task_state) 86 { 87 p_tcb->OSTCBStkPtr = p_tos; 88 p_tcb->OSTCBStat = task_state; 89 } 90 91 void OS_TaskIdle(void *p_arg) 92 { 93 p_arg = p_arg; // No use of p_arg, only for avoiding "warning" here. 94 for(;;) { 95 // OS_ENTER_CRITICAL(); 96 // Nothing to do 97 // OS_EXIT_CRITICAL(); 98 } 99 } 100 101 // void SysTick_Handler(void) 102 // { 103 // // OS_ENTER_CRITICAL(); 104 // // OS_EXIT_CRITICAL(); 105 // } 106 void SysTick_Handler(void) 107 { 108 OS_ENTER_CRITICAL(); 109 if((--TaskTimeSlice) == 0){ 110 TaskTimeSlice = TASK_TIME_SLICE; 111 OSTCBCur = OSTCBNext; 112 OSCtxSw(); 113 OSTaskNext++; 114 while(OSTCBTbl[OSTaskNext].OSTCBStkPtr == (OS_STK*)0) { 115 OSTaskNext++; 116 if(OSTaskNext >= OS_MAX_TASKS) { 117 OSTaskNext = 0; 118 } 119 } 120 OSTCBNext = &OSTCBTbl[OSTaskNext]; 121 TaskTimeSlice = TASK_TIME_SLICE; 122 } 123 TimeMS++; 124 OS_EXIT_CRITICAL(); 125 } 126 127 void SysTickInit(INT8U Nms) 128 { 129 130 OS_ENTER_CRITICAL(); 131 132 TimeMS = 0; 133 TaskTimeSlice = TASK_TIME_SLICE; 134 135 SysTick->LOAD = 1000 * Nms - 1; 136 *Systick_priority = 0x00; 137 SysTick->VAL = 0; 138 SysTick->CTRL = 0x3; 139 OS_EXIT_CRITICAL(); 140 } 141 142 INT32U GetTime(void) 143 { 144 return TimeMS; 145 } 146 147 void delayMs(volatile INT32U ms) 148 { 149 INT32U tmp; 150 tmp = GetTime() + ms; 151 while(1){ 152 if(tmp < GetTime()) break; 153 } 154 } 155 156 void LedInit(void) 157 { 158 RCC->APB2ENR |= 1<<2; 159 RCC->APB2ENR |= 1<<5; 160 //GPIOE->CRH&=0X0000FFFF; 161 //GPIOE->CRH|=0X33330000; 162 163 GPIOA->CRH &= 0xfffffff0; 164 GPIOA->CRH |= 0x00000003; 165 //GPIOA->ODR &= 0xfffffeff; 166 GPIOA->ODR |= 1<<8; 167 168 GPIOD->CRL &= 0xfffff0ff; 169 GPIOD->CRL |= 0x00000300; 170 //GPIOD->ODR &= 0xfffffffd; 171 GPIOD->ODR |= 1<<2; 172 173 //LED1TURN(); 174 LED2TURN(); 175 176 }

現在我們將按照下列過程展開敘述:

初始化OS--》創建任務--》初始化OS時間單位--》初始化LED燈--》啟動OS;

“咦?怎么感覺這些步驟似曾相識呢?”

當然“相識”啦,這個就是main函數的那幾行代碼的意義啊!快看快看,第一行是OSInit(),這個函數到底做了些什么呢?

?

(1)初始化OS:

OSInit將完成以下工作:

I. ?初始化全局變量OSTCBTbl結構體數組;

II. 創建一個“Idle task”;

III.初始化OSTCBCur和OSTCBNext。

1 void OSInit(void) 2 { 3 INT8U i; 4 OS_ENTER_CRITICAL(); 5 for(i = 0; i < OS_MAX_TASKS; i++) { 6 OSTCBTbl[i].OSTCBStkPtr = (OS_STK*)0; 7 OSTCBTbl[i].OSTCBStat = TASK_STATE_CREATING; 8 } 9 OSInitTaskIdle(); 10 OSTCBCur = &OSTCBTbl[0]; 11 OSTCBNext = &OSTCBTbl[0]; 12 OS_EXIT_CRITICAL(); 13 }

?

首先是第4行,就是進入“臨界區”啦,也就是關中斷。

接著是第5~8行的循環,其實就是初始化全局變量OSTCBTbl這個結構體數組,關鍵是這個結構體的指針OSTCBStkPtr,它之后會指向每個任務對應的堆棧,

此處全部初始化為“0”。當任務被創建時,它就會指向實際的內存地址,作為任務的堆棧:

第9行,創建一個“Idle Task”。此處不繼續深挖它,我們后面會深講創建一個任務的具體過程,之后就自然能看懂該函數的具體內容了。

此處要有一個概念,也就是我們在這里已經創建了一個”Task“了,即便我們后來一個“task”也不創建,CPU也會執行“Idle task”的。

然后是第10~11行,使OSTCBCur和OSTCBNext都指向“Idle task”,OSTCBCur指“OS Task Control Block Current”,

OSTCBNext當然是指“OS Task Control Block Next”咯,這兩個指針是后來用于進行“任務切換”的,不知你可否體會呢^_^?

最后是第12行,離開臨界區,也就是開中斷。

?

(2)創建任務:

OSTaskCreate會完成以下工作:

I. ? 找到一個“空閑的”OSTCBTbl;

II. ?初始化參數“p_tos”所指向的內存,并將其作為任務堆棧,最重要的是,使堆棧記錄參數task所指向的函數的入口地址;

III. 設置新的OSTCBTbl的狀態為TASK_STATE_CREATING

(這個狀態變量在本OS中算是個bug,但好在沒有用到這個變量,所以就暫且沒管,所以你也可以暫且不管)

1 void OSTaskCreate(void (*task)(void *p_arg), 2 void *p_arg, 3 OS_STK *p_tos) 4 { 5 OS_STK * tmp; 6 INT8U i = 1; 7 OS_ENTER_CRITICAL(); 8 while(OSTCBTbl[i].OSTCBStkPtr != (OS_STK*)0) { 9 i++; 10 } 11 tmp = OSTaskStkInit(task, p_arg, p_tos); 12 OSTCBSet(&OSTCBTbl[i], tmp, TASK_STATE_CREATING); 13 OS_EXIT_CRITICAL(); 14 }

不得不提醒“小白兔們”,現在我們已經來到了這座OS之山最險峻的地方了!!!

如果實在堅持不下去了,就先休息休息。有時候,就差那么點“心領神會”,施主若是與OS有緣,那么緣分總會來的。

首先講第8~9行的循環是什么意思:

在“OS初始化”中,我們把OSTCBTbl這個結構體數組的OSTCBStkPtr指針都初始化成了“0”,當然這些被初始化的“OSTCBTbl”都是“空閑的”,也就是沒有被分配給具體任務,所以它指向堆棧地址的指針肯定是“0”,如果不是空閑的,它就應當指向具體的堆棧地址。此處循環的跳出條件就是找到某個OSTCBTbl的堆棧指針為“0”,也就是找到空閑的OSTCBTbl,此時的“i”為其偏移量。記住,“我們要創建一個新task,所以我們就需要一個空閑的OSTCBTbl來記錄這個task的堆棧信息。”這樣就能明白這個循環的目的了。

接著是第11行,也就是最難的地方了。

“OSTaskStkInit”,就是這個函數,它會完成以下工作:

I. ? 因為參數p_tos指向堆棧棧頂(Pointer Top Of Stack),所以我們要將其依次遞減,并初始化其下的一段內存中的內容。

OSTaskCreate(Task1, (void*)0, (OS_STK*)&Task1Stk[TASK_STACK_SIZE-1]);

這是main函數調用的部分,這個參數指向的正是棧頂!

1 OS_STK* OSTaskStkInit(void (*task)(void *p_arg), 2 void *p_arg, 3 OS_STK *p_tos) 4 { 5 OS_STK *stk; 6 stk = p_tos; 7 8 *(stk) = (INT32U)0x01000000L; // xPSR 9 *(--stk) = (INT32U)task; // Entry Point 10 11 // Don't be serious with the value below. They are of random 12 *(--stk) = (INT32U)0xFFFFFFFEL; // R14 (LR) 13 *(--stk) = (INT32U)0x12121212L; // R12 14 *(--stk) = (INT32U)0x03030303L; // R3 15 *(--stk) = (INT32U)0x02020202L; // R2 16 *(--stk) = (INT32U)0x01010101L; // R1 17 18 // pointer of the argument 19 *(--stk) = (INT32U)p_arg; // R0 20 21 // Don't be serious with the value below. They are of random 22 *(--stk) = (INT32U)0x11111111L; // R11 23 *(--stk) = (INT32U)0x10101010L; // R10 24 *(--stk) = (INT32U)0x09090909L; // R9 25 *(--stk) = (INT32U)0x08080808L; // R8 26 *(--stk) = (INT32U)0x07070707L; // R7 27 *(--stk) = (INT32U)0x06060606L; // R6 28 *(--stk) = (INT32U)0x05050505L; // R5 29 *(--stk) = (INT32U)0x04040404L; // R4 30 return stk; 31 }

?關鍵是為何要這樣初始化呢?第一次看的話,先從下文找點感覺,看完全部后,還需回來體味體味。

首先,參照《Cortex-M3權威指南(中文版)》P135,表9.1

?

?

中斷發生時,CPU會將以上寄存器按上述順序壓入PSP中,當我們只有一個main任務需要執行時,PSP就足夠幫我們保留現場的了,以至于在中斷返回時,從PSP去取出先前的main任務的寄存器內容就可以了(你是不是能看出上表與我們的函數內容的對應關系呢?)。

但是“多任務”當然就不止一個main任務了,假設我們有3個任務,task1,task2,task3:

task1--》task2;PSP會保留task1的寄存器信息;

task2--》task3;PSP會再保留task2的寄存器信息,但是task1的寄存器信息就被覆蓋掉了!

task3-----????-----task1

那么接下來就悲劇了。

當然,這個函數只是先給以后會用到的堆棧初始化,至于具體值并不重要,除了第8、9、19行:

第8行指定的是一個程序正常運行時,狀態寄存器PSR該有的值;

第9行指定的是任務的入口地址。這個當然重要啦!因為我們的任務就是這個地址所指向的函數。

第19行指定的是任務函數參數的所在地址,因為我們一直賦值為“0”,所以暫且沒有太多意義。

至于其他初始化的值,比如

(INT32U)0x08080808L;
這都無關緊要,隨你怎么設置都行。

?最后第30行,函數返回堆棧指針,該指針現在指向地址的內容為0x04040404L,依照注釋,就是以后存儲R4寄存器內容的地址。

?回到 OSTaskCreate函數第12行,它使得先前找到的OSTCBTbl不再“空閑了”。

?

(3)初始化SysTick中斷和LED:

在main函數中

SysTickInit(5);
LedInit();

都是與硬件相關的,唯一需要說明的是“SysTickInit(5)”當中的“5”沒有太多含義,只是越大,中斷發生的時間間隔就越大,具體依照硬件時鐘而定。

?

(4)main函數最后一行:

OSStart()

該函數一旦執行,將永不返回。注意,接下來我們要和匯編交手了,從匯編部分(os_cpu_a.asm)第31行開始。

這里要請大家注意,每行匯編語言后都有對應的C語言注釋,對照著看更容易理解。

第32行:

CPSID I

參照《Cortex-M3權威指南(中文版)》P42,該命令即為關中斷,相當于給PRIMASK寫“1”。

(注:CPSID指的是屬于CPS(Control Processor State)指令的Interrupt Disable指令)

第36~38行:

LDR R0, =NVIC_SYSPRI14

LDR R1, =NVIC_PENDSV_PRI

STRB R1, [R0]

設置PendSV中斷優先級。

第42~43行:

LDR R4, ?=0x0

MSR PSP, R4

初始PSP寄存器,使之為0。這個“0”表示的是我們的OS才啟動,還沒有任務被運行,后面很快就有說明。

第46~48行:

LDR R4, =NVIC_INT_CTRL

LDR R5, =NVIC_PENDSVSET

STR R5, [R4]

觸發PendSV中斷。

“什么!觸發中斷了!哎呀哎呀,怎么辦?中斷函數在哪兒?”

小白兔請先別著急,由于先前我們已經關掉中斷了,所以程序還會繼續往下執行,直到我們再開啟中斷。

由于我們真正的目的就是要開啟PendSV中斷,所以下一步我們要開中斷啦!

第51行:

CPSIE I

開中斷。由于PendSV被觸發了,所以接下來CPU跳到PendSV的中斷處理函數處執行。也就是os_cpu_a.asm的第59行。

第60行是關中斷。

第62~63行:

MRS R0, PSP

CBZ R0, PendSV_Handler_NoSave

比較PSP是否為0,若是0就跳轉到PendSV_Handler_NoSave處執行,由于OSStart先前將PSP初始化為0,所以就直接調到PendSV_Handler_NoSave處執行咯。

第104~107行

LDR R0, =OSTCBCur

LDR R1, =OSTCBNext

LDR R2, [R1]

STR R2, [R0]

將OSTCBNext所指向地址的前4個字節,賦給OSTCBCur所指向地址的前4個字節,而這4個字節正是兩個指針所指向結構體的OSTCBStkPtr變量!

這段代碼做的就是將下個任務的堆棧指針(OSTCBNext)賦給當前任務的堆棧指針(OSTCBCur),因為PendSV中斷所做的事情就是進行任務切換。

我們知道,一開始OSTCBCur和OSTCBNext一開始都是指向“Idle Task”的,所以下一個要運行的任務就是“Idle Task”。

第109行:

LDR R0, [R2]

將所要切換的堆棧指針地址賦給R0。

第112~134行:

LDR R4, [R0]

ADD R0, R0, #0x4

LDR R5, [R0]

ADD R0, R0, #0x4?

……

LDR R11 , [R0]

ADD R0, R0, #0x4

將堆棧指針R0所指向的堆棧內容賦給R4~R11。問個問題,“這些值都是多少你知道嗎?”

“好了,這時候你是不是想問個問題,R0~R3怎么不給它們也賦值呢?”

回到先前那個要大家需要體味的地方

參照《Cortex-M3權威MSR PSP, R0指南(中文版)》P135,表9.1

還有接下來第136行:

MSR PSP, R0

我只說一句:CPU會自動從PSP保存和加載8個量至R0~R3,R12,LR,程序入口(這個是理解的關鍵)和xPSR,我們無需動手。

第140行:

ORR LR, LR, #0x04

參照《Cortex-M3權威MSR?PSP, R0指南(中文版)》P139,這行是為了在回到任務后,確保繼續使用PSP堆棧。

第141行開中斷,此時一般還不會有中斷介入,但我們要記住,現在我們還處在PendSV中斷中,第142從PendSV中斷返回,返回后CPU則去新的任務處執行了。

?

?

讓我們再回到OS_cpu_a.asm的第62~63行,當PSP不為零,也就是已經有任務運行了,那么我們就需要先保存這個將被切換出去的任務的寄存器信息。

現在PSP指向的堆棧地址如圖所示:

?

完成第62~66行命令之后,我們得到:

該圖中有兩個”R0",并不是很得體,我還是解釋一下,左邊的R0指的是PendSV中斷下正在使用的R0寄存器,右邊的R0指的是被中斷的任務的R0寄存機所存儲的值。

保護現場,保護現場啦!將R4~R11全部存儲起來。

“為什么R0~R3不用保護呢?”

因為……你看啊,其實CPU自己已經在PendSV中斷發生時,“擅自”把它們存儲了,就在上圖啊!

完成第69~91行命令之后,再將R0減去0x20,堆棧就變成這樣啦:

然后是第99~101行:

把R0的值賦給OSTCBCur的OSTCBStkPtr指針,這下我們就放心了,所有有關任務的信息都被存放在了相應的OSTCBTbl中了。

接下來就是該切換進入新任務了。

?

(5)誰來觸發PendSV,實現切換任務

如果到此為止,那么CPU只會一直沒完沒了的執行第一個任務:Idle Task。OSStart確實完成了一次任務切換,但也僅僅就“一次”。

main函數已經沒得指望了,它早就撒手不管了,那么誰來再次觸發PendSV,從而執行Task1、Task2呢?

別忘了,我們還有一個關鍵角色沒有登場呢:SysTick中斷。

讓我們回到myos.c的106行的SysTick的中斷服務函數:SysTick_Handler。

這個函數邏輯很簡單:每發生一次SysTick中斷,就將TimeTaskSlice遞減,當TimeTaskSlice為0時,就進行切換任務的工作,并將TimeTaskSlice的值還原。

每一次中斷發生,TimeMS都會加1,從而記錄整個系統的時間。

第111行:

OSTCBCur = OSTCBNext;

將當前任務指針指向下一個任務。

第112行:

OSCtxSw();

這是個匯編實現的函數,在os_cpu_a.asm的第144~150行。

非常簡單,它就是在觸發PendSV中斷!!!

當然,這個觸發不會立即發生,因為現在還處于關中斷狀態。

第113行:

OSTaskNext++;

之前忘記介紹了,這是個全局整型變量,用來記錄下一個任務的偏移量的。由于我們的任務沒有優先級,只是輪換執行,所以將OSTaskNext向后偏移一個就行了。

但是,如果我們偏移到了最后一個任務怎么辦呢?我們得從第一個任務重新開始才是,所以就有了第144~149的循環部分。

第120行:

OSTCBNext = &OSTCBTbl[OSTaskNext];

設置新的下一個要運行的任務的任務指針。

第121行有點多余,不要也行。

第124行開中斷,這行命令執行完之后,PendSV就會被觸發,接著就是去執行新的任務咯。

?

至此,關于本OS的關鍵代碼部分就解析完畢了。

真心希望大家多提意見,鄙人將感激涕零!

ansersion@sina.com

轉載于:https://www.cnblogs.com/ansersion/p/4328800.html

總結

以上是生活随笔為你收集整理的一步步学习操作系统(1)——参照ucos,在STM32上实现一个简单的多任务(“啰里啰嗦版”)...的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

九九免费精品视频在线观看 | 久久午夜精品影院一区 | 日韩资源在线 | 欧美一区二区三区在线播放 | 黄a在线看 | 超级碰碰碰视频 | 在线观看精品黄av片免费 | 久久精品艹 | 欧美巨大荫蒂茸毛毛人妖 | 免费91在线| 国产91精品看黄网站 | 中国老女人日b | 韩日电影在线观看 | 91精品在线看 | www免费黄色 | 91精品国产高清自在线观看 | 92国产精品久久久久首页 | 日韩免费在线看 | 日日夜夜人人天天 | 午夜精品久久久久久久久久久久 | 久久久久久久久久久久电影 | 欧美另类成人 | 免费看色视频 | 在线看黄网站 | 草在线| 久久99久久99精品免观看软件 | 日韩欧美在线视频一区二区三区 | 日韩在线视频国产 | 国产精品精品国产色婷婷 | 中文字幕黄色 | 成人午夜电影在线 | 久久久受www免费人成 | 欧美极品xxx| 久久久久99精品成人片三人毛片 | 亚洲一区免费在线 | 国产成人精品在线观看 | 亚洲综合激情小说 | 97视频人人| 免费91麻豆精品国产自产在线观看 | 91精品天码美女少妇 | 午夜久久美女 | 香蕉影院在线播放 | 五月婷婷网站 | 亚洲 欧洲 国产 日本 综合 | 午夜精品成人一区二区三区 | 国产第一页在线观看 | 偷拍久久久 | 九九九热 | 国产视频手机在线 | 免费精品国产 | 亚洲成a人片综合在线 | 精品在线视频观看 | 欧美精品久久久久性色 | 国内少妇自拍视频一区 | 激情校园亚洲 | 久久久久久久久久久久99 | 中文字幕av最新 | 五月天久久久久 | 亚洲影院色 | 丝袜av一区 | 美女av免费| 麻豆av电影| 国产免费专区 | 精品av在线播放 | 亚洲艳情 | 91精品国产99久久久久久久 | 国产成人一区二区啪在线观看 | 免费国产亚洲视频 | 九九九九免费视频 | 在线高清| 玖玖在线播放 | 久久99精品国产99久久6尤 | 久草视频在线资源站 | 中文字幕在线播放一区 | 九色在线| 五月婷婷操 | 中文字幕一区二区在线播放 | 久久久午夜剧场 | 亚洲最新视频在线播放 | av成人在线看 | 精品国产一区二区三区久久 | 中文字幕第 | 成人91视频 | 91最新网址在线观看 | 最近日本韩国中文字幕 | 久久久在线观看 | 亚洲激情综合 | 欧美成天堂网地址 | 92中文资源在线 | 亚洲精品久久久久久中文传媒 | 久久综合狠狠综合 | 最近中文字幕完整高清 | 久久五月婷婷丁香社区 | 日本中文字幕在线免费观看 | 四虎国产永久在线精品 | 日韩电影中文字幕在线 | 国产91粉嫩白浆在线观看 | 日韩精品视频一二三 | 日本精品久久 | 久久深夜 | 国产高清不卡一区二区三区 | 国产精品高清免费在线观看 | 欧亚久久 | 国产日韩欧美在线观看视频 | 99在线视频观看 | 国产成人福利在线 | 国产青草视频在线观看 | 国产视频日本 | 香蕉成人在线视频 | 在线观看视频一区二区三区 | 看黄色91| 欧美俄罗斯性视频 | 99热最新 | 99国产一区二区三精品乱码 | 99热在线国产 | 成人香蕉视频 | 在线观看中文字幕 | 69国产盗摄一区二区三区五区 | 久99久久| 色视频网址 | 天天干天天射天天插 | 国产精品国产三级国产专区53 | 五月婷婷网站 | 欧美精品乱码久久久久 | 成年免费在线视频 | 亚洲综合激情 | 久久国产精品二国产精品中国洋人 | 国产精品精品久久久久久 | 一区二区不卡在线观看 | 亚洲精品人人 | 久久国产一区二区三区 | 91女神的呻吟细腰翘臀美女 | www操操操 | 欧美性做爰猛烈叫床潮 | 成人av在线网址 | 欧美另类重口 | 91少妇精拍在线播放 | 99色在线 | 日韩精品欧美视频 | 麻豆国产精品视频 | 国产精品麻豆欧美日韩ww | 久久激情视频 久久 | 伊人精品在线 | 久久精品中文字幕免费mv | 国产在线 一区二区三区 | 久久公开视频 | 亚洲 欧美 变态 国产 另类 | 91成熟丰满女人少妇 | 国产精品 久久 | 国产午夜三级一区二区三 | 日韩美av在线 | 天天操天天干天天玩 | 五月婷婷黄色网 | 特级西西444www高清大视频 | 国产麻豆精品久久 | 91九色pron| 五月激情五月激情 | 亚洲免费av片| 色福利网 | 久久在线免费观看 | 91视频啪 | 日韩欧美一区二区三区黑寡妇 | 久色网| 精品久久国产一区 | 欧美最猛性xxx | 国产99爱| 狠狠狠干狠狠 | 亚洲黄色一级电影 | 丰满少妇在线观看网站 | 国产在线播放一区二区三区 | 中文字幕电影高清在线观看 | 久久久久久久久久久久久久电影 | 99久久精品国产观看 | 国产精品久久久久久一区二区三区 | 亚洲成年人在线播放 | 国产精品女人网站 | www.777奇米 | 免费男女羞羞的视频网站中文字幕 | 婷婷视频导航 | 精品久久福利 | 黄色在线观看免费 | av超碰免费在线 | 一区二区三区四区不卡 | 久久深夜福利免费观看 | 综合激情婷婷 | 精品国产免费看 | 综合久久久久久久 | 一区二区三区免费在线 | 四季av综合网站 | 91免费网| 九九涩涩av台湾日本热热 | 在线观看午夜 | 1024手机基地在线观看 | 欧美性久久久 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 美女激情影院 | 精品一区二区在线免费观看 | 日韩激情在线视频 | 亚洲精品国产精品久久99热 | 婷婷av电影 | 久久久精选 | 亚洲精品高清一区二区三区四区 | 久久久久久久久精 | 久久免费电影网 | 国内精品视频在线 | 婷香五月 | 日本久久综合视频 | 久草精品网 | 亚洲三级在线免费观看 | 91久久久久久国产精品 | 国产午夜精品在线 | 91精品一区在线观看 | 一本一本久久a久久精品综合妖精 | 外国av网 | 国产精品毛片一区二区 | 久久污视频 | 久久久久久久久久久久久久av | 亚洲六月丁香色婷婷综合久久 | 六月色 | 免费韩国av | 狠狠干狠狠色 | 成年人看片网站 | 美女网站在线观看 | 中文在线中文资源 | 三级黄色网络 | 操操操com| 黄色一二级片 | 国产亚洲久一区二区 | 国产精品毛片一区二区在线看 | 狠狠操综合 | 日日操日日插 | 在线观看www视频 | 深夜免费福利 | 国产在线观看一 | 国产美女免费观看 | 天天舔天天射天天操 | 免费在线观看av的网站 | 国产精品久久久久久久免费大片 | 久久av网址 | 日韩成人在线一区二区 | 国产一区二区手机在线观看 | 欧美日韩另类在线 | 久久午夜网 | 亚洲国产视频直播 | 国产日韩欧美视频 | 国产精品久久久久久久久久久久 | 日韩av资源站 | 欧美日韩视频 | 97超碰人 | 色婷婷在线视频 | 日韩一区二区三区免费视频 | 成人宗合网 | 91最新视频在线观看 | 午夜色站| 91av成人| 96精品高清视频在线观看软件特色 | 亚欧洲精品视频在线观看 | 99精品视频观看 | 成人午夜久久 | 久久久亚洲麻豆日韩精品一区三区 | 欧美极品久久 | 91在线成人| 狠狠干成人 | 99色精品视频 | 欧美最猛性xxxx | 日韩一区二区免费在线观看 | 国产精品手机在线播放 | 久久人人97超碰精品888 | 91麻豆精品国产自产在线 | 九九色网| 国产黄色av影视 | 精品一区二区三区久久 | 国语久久 | 91av视频免费观看 | 国产欧美精品一区二区三区 | 亚洲精品激情 | 精品久久久影院 | 欧美亚洲国产精品久久高清浪潮 | 在线观看久久 | 一区二区三区视频在线 | 天天婷婷 | 色干干| 91色偷偷 | av看片在线观看 | 黄色一二级片 | 91精品一区二区三区蜜臀 | 深爱激情五月综合 | 在线视频福利 | 亚洲精品久久久久久中文传媒 | 六月婷操| 久久久国产视频 | 99婷婷狠狠成为人免费视频 | 超碰免费久久 | 天天射天天射天天 | 色免费在线 | 日韩在线免费视频观看 | av永久网址 | 成人午夜电影免费在线观看 | 欧美性色19p | 香蕉视频最新网址 | 久久精品视频免费观看 | 亚州免费视频 | 国产一区二区三区在线 | 午夜精品久久一牛影视 | 国产精品久久久久婷婷二区次 | 天天综合网 天天 | 日韩在线观看网站 | 狠狠干.com| 国产成人精品三级 | 三级在线国产 | 91大神精品视频在线观看 | 91精品视频免费看 | 在线观看视频在线观看 | 精品国内自产拍在线观看视频 | 久久人网| 成人9ⅰ免费影视网站 | 国产精品久久久久婷婷 | 成人久久久久久久久久 | 中文国产在线观看 | 久久九九国产视频 | 91污污视频在线观看 | 久久久国产精品一区二区中文 | 日韩xxxbbb| 久久精品国产免费看久久精品 | 午夜av大片 | 激情五月婷婷激情 | 91av视频在线观看免费 | 精品嫩模福利一区二区蜜臀 | 成人午夜片av在线看 | 成人国产精品av | 亚洲精品在线一区二区三区 | 亚洲激情视频在线 | 伊人伊成久久人综合网小说 | 精品人人人人 | 五月婷婷在线视频观看 | 婷婷色婷婷| 麻豆成人在线观看 | 在线精品一区二区 | 视频在线精品 | 久久精品7 | 久草视频免费在线播放 | 国产午夜麻豆影院在线观看 | 国产一性一爱一乱一交 | 日韩欧美高清一区二区三区 | 不卡av免费在线观看 | 99久久精品国产网站 | 91丨九色丨高潮 | 国产精久久久 | 国产精品久久网站 | 国产精品igao视频网网址 | av成人资源 | 天天操导航| 高清有码中文字幕 | 99久久这里只有精品 | 亚洲精品男人天堂 | 999久久久免费精品国产 | 国产精品久久久久久久久软件 | 成人毛片100免费观看 | 黄色av在| 国产免费人成xvideos视频 | 日韩电影一区二区在线 | 国产精品视频免费看 | 亚洲精品视频网 | 色综久久 | 天天视频色版 | 在线天堂日本 | 久久久99精品免费观看app | 久久在现视频 | 2018亚洲男人天堂 | 国产午夜在线观看视频 | 欧美性久久久久久 | 91久久精品日日躁夜夜躁国产 | 99视频久| 在线观看国产区 | 九九久久成人 | 一区二区网 | 欧美成人精品欧美一级乱黄 | 国产成人91 | 国产精品女同一区二区三区久久夜 | 日韩精品免费在线视频 | 激情五月在线 | 国产一区欧美在线 | 中文字幕日韩精品有码视频 | 在线中文日韩 | 激情www| 五月天婷婷狠狠 | 狠狠色噜噜狠狠狠狠2022 | 天天干天天干天天 | 四虎国产精品成人免费影视 | 欧美天天射| 97在线视频免费观看 | 69精品人人人人 | 久草com| 国产视频综合在线 | 在线免费观看麻豆 | 国产成人精品一二三区 | 91成人在线观看喷潮 | 国精产品999国精产 久久久久 | 欧美日韩69 | 91社区国产高清 | 久久乐九色婷婷综合色狠狠182 | 狠狠躁夜夜av | 日韩av免费一区 | 视频在线观看入口黄最新永久免费国产 | 亚洲精品在线免费观看视频 | 午夜免费福利视频 | 国产亚洲小视频 | 黄色的视频网站 | 国产精品久久99综合免费观看尤物 | 免费高清男女打扑克视频 | 亚洲成人精品在线 | 00av视频| 国产精品免费成人 | 国产精品剧情在线亚洲 | 免费三级骚 | 激情五月在线视频 | 精品视频资源站 | 免费看在线看www777 | 久久久 激情 | 黄色三级免费片 | 国产精品午夜久久 | 国产福利91精品一区 | 国产亚洲精品久久久久久移动网络 | 国产一区欧美一区 | 一区二区网| 国产又粗又猛又爽 | 日韩精品三区四区 | 超碰人人在线观看 | 91视频久久久久 | 在线欧美小视频 | 波多野结衣在线观看一区二区三区 | 色噜噜狠狠狠狠色综合久不 | 国产精品免费视频观看 | 婷婷午夜激情 | 国产精品第一页在线 | 天天干天天操天天搞 | 国产成人一区二区三区影院在线 | 国产欧美在线一区二区三区 | 特级西西444www高清大视频 | 成人久久毛片 | 激情xxxx| 欧美激情精品一区 | 久久久久综合视频 | 久久免费看a级毛毛片 | 日韩欧美国产精品 | 91色在线观看 | 国产黄色精品 | 天天干夜夜夜操天 | 99视频在线看 | 日本三级久久久 | 日韩精品播放 | 亚洲精品视频在线播放 | 亚洲视频在线观看 | 欧美一性一交一乱 | 中文字幕在线观看一区二区三区 | 国产五月色婷婷六月丁香视频 | 麻豆国产露脸在线观看 | 国产一级特黄毛片在线毛片 | 免费午夜av | 国产午夜精品一区二区三区 | 中文伊人 | 一本—道久久a久久精品蜜桃 | 国产精品欧美久久久久久 | 伊人射| 超碰在线97免费 | 美女网站黄在线观看 | 中文字幕国语官网在线视频 | 久久久亚洲影院 | 亚洲视频免费在线观看 | www五月天婷婷| 国产黄在线免费观看 | 一本一本久久a久久精品综合小说 | 久久久精品一区二区三区 | 99精品视频一区 | 午夜精品久久久久久久99水蜜桃 | 国内精品亚洲 | 国产精品视频大全 | 99精品一区二区三区 | 久久久国产精品成人免费 | 国产老太婆免费交性大片 | 亚洲激情综合 | 91片网 | 99精品视频在线播放观看 | 精品女同一区二区三区在线观看 | 欧美性高跟鞋xxxxhd | 91免费网 | 国产亚洲久一区二区 | 在线日韩中文字幕 | 久久久久国产精品免费免费搜索 | 黄色成人在线观看 | av不卡免费在线观看 | 国产精品手机在线观看 | www视频免费在线观看 | 99在线精品视频 | 久久国产精品影视 | 成年人在线电影 | 久操伊人| 在线色亚洲 | 久久99精品国产一区二区三区 | 97视频在线免费 | 欧美性色综合网 | 99电影| 四虎8848免费高清在线观看 | 九九免费在线观看视频 | 日本在线观看一区 | 狠狠干在线| 久久免费黄色网址 | av中文字幕在线免费观看 | 在线观看aa| 婷婷久久网 | 手机成人免费视频 | 日本久久精品视频 | av免费黄色 | 天天天色 | 亚洲 中文 欧美 日韩vr 在线 | 天天做日日做天天爽视频免费 | 精品亚洲va在线va天堂资源站 | 天天色天天色天天色 | 国产精品国产三级国产aⅴ入口 | 五月亚洲综合 | 欧美性性网 | 9992tv成人免费看片 | 亚洲国产中文字幕 | 91插插插免费视频 | 国产成人精品一区二 | 人人插人人舔 | 国产精品大片 | 日本三级不卡视频 | 91av色| 免费黄色av电影 | 国产人免费人成免费视频 | 天堂av在线免费观看 | 午夜久久久影院 | 91麻豆精品国产自产在线 | 亚洲精品日韩在线观看 | 天天色天天爱天天射综合 | 中文字幕视频网站 | 国产日韩精品一区二区 | 久久经典国产 | 日韩一二三| 国产高清99 | 国产精品毛片一区二区三区 | 国产va饥渴难耐女保洁员在线观看 | 国产精品99久久久精品免费观看 | 午夜精品一二三区 | 国产欧美在线一区二区三区 | 成 人 黄 色 免费播放 | 2019精品手机国产品在线 | 精品一区二区久久久久久久网站 | 日日夜夜骑 | 波多野结衣在线中文字幕 | 国产原厂视频在线观看 | 在线黄色观看 | 色综合夜色一区 | 美女免费视频网站 | 国产特级毛片aaaaaa高清 | 五月天久久婷婷 | 欧美激情综合色综合啪啪五月 | 久久婷婷一区二区三区 | 欧美日韩69 | 中文字幕在线观看国产 | 在线观看亚洲精品视频 | 丝袜制服天堂 | 奇米网8888 | 欧美a免费| 精品亚洲va在线va天堂资源站 | 亚洲少妇久久 | 一区二区三区免费在线播放 | 亚洲精品字幕在线观看 | av免费在线观看网站 | 久精品视频免费观看2 | 久久不卡免费视频 | av再线观看 | 久久久久免费精品视频 | 中文字幕麻豆 | 在线国产一区二区三区 | 日韩免费在线 | 97超碰福利久久精品 | 成人国产综合 | 日日操操操 | 中文字幕区 | 亚洲一区二区视频在线 | 免费在线观看日韩视频 | 久久爱资源网 | 日日干av | 黄色aaa毛片 | 色网站在线 | 日韩网站在线 | 成人av片在线观看 | 韩国av一区二区三区在线观看 | 不卡中文字幕在线 | 91片网 | 国内外成人在线视频 | 久久久免费毛片 | 日本久久久久久久久久久 | 免费特级黄毛片 | 波多野结衣视频一区二区 | 激情久久五月天 | 日日操狠狠干 | www.久久精品视频 | 成人一级免费视频 | 五月婷婷丁香 | 亚洲自拍av在线 | 国产又粗又猛又色又黄网站 | 日韩视频免费观看高清 | 日韩免费视频在线观看 | 中文av字幕在线观看 | 久久久精品电影 | 午夜 在线 | 一区二区久久久久 | 午夜神马福利 | 国产直播av | 久久久九九 | 亚洲国产合集 | 国产高清免费视频 | a在线观看视频 | 欧美久久久久久久久久久 | 四虎在线观看 | 亚洲六月丁香色婷婷综合久久 | 久久精品中文字幕 | 日本久久免费电影 | 免费进去里的视频 | 免费av网站在线 | 久章操 | 91毛片视频| 丁香六月激情婷婷 | 日日夜夜精品免费观看 | 亚洲欧美国产视频 | 国产原创av片 | 久久国语 | 久久99电影 | 四虎影院在线观看av | 久久毛片高清国产 | 韩日av一区二区 | 97超碰人人网 | www日韩精品 | 在线你懂的视频 | 国内精品久久久久久久久 | 黄色毛片在线观看 | 国产在线最新 | 最近日本mv字幕免费观看 | 国产区在线视频 | 日韩精品在线免费播放 | 日本夜夜草视频网站 | 天天综合五月天 | 亚洲国产精品va在线看黑人动漫 | 色婷婷狠| 欧美一区二区在线看 | 日韩精品三区四区 | 国产精品成人国产乱一区 | 99精品偷拍视频一区二区三区 | 久久这里只有精品视频99 | 久久久夜色| 91大神视频网站 | 中国一级特黄毛片大片久久 | 免费看毛片网站 | 99精彩视频在线观看免费 | 久久激情片 | 日本黄色大片免费看 | 国产性xxxx | 欧美色婷婷 | 成人av在线看 | 免费看的黄色小视频 | 国产高清不卡 | 国产黑丝一区二区 | 亚洲精品456在线播放 | 一区二区三区四区精品视频 | 色综合久久五月天 | 日韩免费电影网站 | 亚洲高清精品在线 | 五月天婷亚洲天综合网鲁鲁鲁 | 日韩精品久久久久 | 狠狠色香婷婷久久亚洲精品 | 精品国产一区在线观看 | 亚洲精品视频中文字幕 | 欧美成人h版 | 亚洲免费在线播放视频 | 五月婷婷在线观看 | 亚洲精品国产精品国自产观看浪潮 | 国产高清不卡av | 成年人天堂com | 日韩免费高清在线 | 久久视频这里有久久精品视频11 | 久久免费观看少妇a级毛片 久久久久成人免费 | 日日夜夜天天久久 | 国产视频一区在线 | 狠狠色噜噜狠狠狠狠 | 四虎影视成人永久免费观看视频 | 97精品国产97久久久久久粉红 | 中文字幕一区二区三区在线播放 | 国产成人一区二区三区在线观看 | 日韩欧美一级二级 | 懂色av懂色av粉嫩av分享吧 | 日本黄色大片免费看 | 国产精品麻豆99久久久久久 | 国产三级精品在线 | 亚洲精品美女视频 | 国产精品 国内视频 | 亚洲国产精品影院 | 国产精品乱码在线 | 九九导航| 激情亚洲综合在线 | 亚洲极色 | 91av官网| 又黄又爽又刺激视频 | 国产精品欧美久久久久无广告 | 久久精品视频18 | 久久男人视频 | 激情五月看片 | 成年人在线视频观看 | 这里有精品在线视频 | 欧美精品在线一区二区 | 日韩av高清| 亚洲欧美偷拍另类 | 日本黄色片一区二区 | 天天爽人人爽夜夜爽 | 成人在线一区二区 | 五月婷婷综合网 | 在线探花 | 中文字幕乱码一区二区 | 一级α片免费看 | 人人添人人澡人人澡人人人爽 | 少妇搡bbbb搡bbb搡忠贞 | 国产香蕉久久精品综合网 | 精品久久久久久久久久久院品网 | 看毛片网站 | 中文字幕免费在线 | 中文字幕一区二区三区四区视频 | 91精品在线看 | 久久观看最新视频 | 91香蕉亚洲精品 | 久久国产精品久久精品国产演员表 | 亚洲国产小视频在线观看 | 中国一级片免费看 | 日本爱爱片 | 五月天六月丁香 | 亚洲精品午夜视频 | 涩涩网站在线播放 | 欧美成人精品三级在线观看播放 | 久久国产一区 | 在线 影视 一区 | 99精品视频免费观看 | 黄色av一区二区 | 亚州精品成人 | 欧美精品久久久久久久久久久 | 国产xvideos免费视频播放 | 久久久久婷 | 日日夜夜天天久久 | 中文字幕欲求不满 | 丁香九月激情综合 | 亚洲精区二区三区四区麻豆 | 成人免费精品 | 丁五月婷婷 | 自拍超碰在线 | 免费又黄又爽视频 | 欧美激情视频在线免费观看 | 91超级碰碰 | 国产特级毛片aaaaaa毛片 | 中文字幕 第二区 | 99人成在线观看视频 | 免费韩国av| 日本在线观看黄色 | 日本乱视频 | 黄色毛片视频免费观看中文 | 丁香六月婷婷开心 | 五月天久久精品 | 免费在线观看污网站 | av黄免费看 | 国产成人精品福利 | 91在线蜜桃臀 | av福利免费 | 在线观看91精品国产网站 | 特级毛片在线 | 日韩一级片网址 | 日韩在线视 | 一级黄色av| 激情婷婷av | 在线观看国产区 | 婷婷丁香国产 | 中文字幕成人 | 五月婷婷久 | 亚洲夜夜综合 | 精品国产乱码一区二区三区在线 | 91九色视频 | 特级毛片网站 | 综合色综合 | 片网址| 中文乱码视频在线观看 | 免费日韩电影 | 国产高清在线免费视频 | 91精品推荐 | 国产美女久久久 | 国产成人精品一区二区三区在线观看 | 99热这里只有精品在线观看 | 波多野结衣久久精品 | 国产精品久久人 | 一区二区在线电影 | 国产女教师精品久久av | 婷婷六月丁 | 免费中文字幕 | 97国产小视频 | 久免费| 亚洲闷骚少妇在线观看网站 | 欧美一区三区四区 | 97国产在线观看 | 97人人网| 免费观看日韩av | 免费在线电影网址大全 | 日韩av福利在线 | 国产亚洲精品久久19p | 国产精品精品久久久久久 | 亚洲视频 在线观看 | 国产一区二区电影在线观看 | 视频精品一区二区三区 | 日韩视频一区二区在线 | 日韩一区在线免费观看 | 天天激情综合 | 久久黄色精品视频 | av中文国产 | 有码中文在线 | 亚洲aⅴ在线观看 | 日韩网站一区二区 | 亚洲视频免费视频 | 国产一区二区免费在线观看 | av网站在线观看免费 | 色播五月激情五月 | 欧美日韩一区三区 | 久久免费在线观看 | 日韩免费看视频 | 91色九色| 午夜成人免费影院 | 一区二区三区高清 | 九九精品视频在线 | 一区二区不卡视频在线观看 | 狠狠狠狠狠狠狠狠干 | 蜜桃传媒一区二区 | 亚洲综合视频在线观看 | 国产综合在线视频 | 奇米四色影狠狠爱7777 | 精品国产1区二区 | 国产97视频在线 | 国产高清无线码2021 | 中文字幕在线视频网站 | 国产精品日韩欧美一区二区 | 成人午夜网址 | wwwww.国产| 97在线观看视频免费 | 中文字幕资源网 | 国产麻豆精品传媒av国产下载 | 中文字幕免费播放 | 国产在线精品播放 | 午夜精品视频一区二区三区在线看 | 少妇精品久久久一区二区免费 | 国产黄色片免费观看 | 色婷五月天 | 97**国产露脸精品国产 | 国产精品九九久久久久久久 | 日韩欧美视频 | 玖操 | 91精品在线观看入口 | 黄p网站在线观看 | 国产aa精品| 日本久久免费电影 | 久久国产品 | 久久久久久免费毛片精品 | 五月天婷婷免费视频 | 免费男女网站 | 亚洲一区视频免费观看 | 少妇bbr搡bbb搡bbb| 激情网综合 | 一区二区三区在线视频111 | 精品999 | 天天干天天操天天入 | 黄色免费在线视频 | 最近日本韩国中文字幕 | 奇米四色影狠狠爱7777 | 国产黄色高清 | 中文字幕精品三级久久久 | 亚洲一级在线观看 | 亚洲黄色软件 | 91中文字幕网 | 免费日韩一区二区三区 | 精品一区二区免费视频 | 久久国产香蕉视频 | 91在线porny国产在线看 | 在线观看黄污 | 亚洲精品男人的天堂 | 福利一区在线 | 激情综合五月婷婷 | www.久久com| 国产精品二区三区 | 免费a一级| 天天操天天干天天爱 | 亚洲aⅴ久久精品 | 狠狠色噜噜狠狠狠 | 国产精品一区二区在线观看免费 | 免费久久99精品国产 | 欧美精品在线观看 | 亚洲一级片在线看 | 特级毛片网站 | 国产一区国产二区在线观看 | 综合久久综合久久 | 丰满少妇一级片 | 久草在线观看视频免费 | 国内精品久久久久 | 天天人人综合 | 日本黄区免费视频观看 | 亚洲免费精品视频 | 成年人免费在线看 | 精品国产自在精品国产精野外直播 | 久久久久久久久久久电影 | 操少妇视频 | 中文国产字幕在线观看 | 日韩精品一区在线观看 | 激情欧美丁香 | 国产精品成人一区二区三区吃奶 | 激情五月在线视频 | 国产精品视频99 | 美女性爽视频国产免费app | 欧美午夜精品久久久久 | 国产精品2020 | 日韩在线视频网站 | 午夜av在线免费 | 天堂网中文在线 | 91精品在线免费视频 | 国产精品久久久久久久久久久杏吧 | 伊人久久婷婷 | 激情综合色图 | 免费看黄的 | 国产精品麻豆三级一区视频 | 在线观看黄色国产 | 福利视频一区二区 | 久久久精品 | 国产资源网站 | 亚洲最新合集 | 日本一区二区免费在线观看 | 在线观看亚洲国产精品 | 91av在线看 | 国产高清久久久久 | 国产污视频在线观看 | 亚洲一区二区三区精品在线观看 | 国产原创在线 | 天堂av在线中文在线 | 久久久久麻豆v国产 | 人人超碰在线 | 亚洲国产中文字幕在线 | 亚洲视频久久久久 | 国产精品久久久久久久久软件 | 久久国精品 | 手机在线看a | 亚洲成人在线免费 | 婷婷黄色片 | 国内精品久久久久影院男同志 | 国产久草在线观看 | 麻豆 videos | 97碰碰碰 | 天天操天天添 | 一区二区三区四区免费视频 | 天天操天天吃 | 成人在线观看资源 | 中文字幕字幕中文 | 国产污视频在线观看 | 91精品国产99久久久久久久 | 国产成人黄色av | 黄色一级大片免费看 | 亚洲精品小视频在线观看 | 激情黄色一级片 | 国产精品区二区三区日本 | 嫩草伊人久久精品少妇av | 十八岁以下禁止观看的1000个网站 | 久久综合狠狠综合久久综合88 | 俺要去色综合狠狠 | 99精品欧美一区二区三区 | 亚洲精品99久久久久久 | 国产精品久久久久久吹潮天美传媒 | 黄网站色成年免费观看 | 国产一区二区在线观看免费 | 欧美在线99 | 国产日韩视频在线观看 | 在线免费中文字幕 | 91色亚洲 | 麻豆国产网站入口 | 久久精品视频在线 | 欧美日韩99| 婷婷在线网站 | 久久久久亚洲精品成人网小说 | 97精品一区| 国产高清视频免费最新在线 | 精品免费视频123区 午夜久久成人 | 国产一卡二卡在线 | 色九九在线 | 精品视频国产一区 | 91精品国产综合久久婷婷香蕉 | 波多野结衣精品在线 | 日本护士三级少妇三级999 | 久久在线免费观看视频 | 久草在线播放视频 | 国产精品不卡在线 |