forked from xuos/xiuos
				
			Support blocking server.
This commit is contained in:
		
							parent
							
								
									9f9e25a98e
								
							
						
					
					
						commit
						3a985252d9
					
				|  | @ -24,7 +24,7 @@ INC_DIR = 	-I$(KERNEL_ROOT)/services/shell/letter-shell \ | ||||||
| 			-I$(KERNEL_ROOT)/services/app  | 			-I$(KERNEL_ROOT)/services/app  | ||||||
| 
 | 
 | ||||||
| ifeq ($(BOARD), imx6q-sabrelite) | ifeq ($(BOARD), imx6q-sabrelite) | ||||||
| all: init test_fs simple_client simple_server shell fs_server test_irq_hdlr test_irq_send readme.txt | bin | all: init test_fs simple_client simple_server shell fs_server test_irq_hdlr test_irq_block test_irq_send readme.txt | bin | ||||||
| else  | else  | ||||||
| all: init test_fs simple_client simple_server shell fs_server test_irq_hdlr readme.txt | bin | all: init test_fs simple_client simple_server shell fs_server test_irq_hdlr readme.txt | bin | ||||||
| endif | endif | ||||||
|  | @ -42,6 +42,10 @@ test_irq_send: test_irq_sender.o usyscall.o libserial.o | ||||||
| 	@${objdump} -S $@ > $@.asm | 	@${objdump} -S $@ > $@.asm | ||||||
| endif | endif | ||||||
| 
 | 
 | ||||||
|  | test_irq_block: test_irq_block.o libserial.o libipc.o session.o usyscall.o libmem.o | ||||||
|  | 	@${ld} ${user_ldflags} -e main -o $@ $^ ${board_specs} | ||||||
|  | 	@${objdump} -S $@ > $@.asm | ||||||
|  | 
 | ||||||
| test_irq_hdlr: test_irq_handler.o libserial.o libipc.o session.o usyscall.o libmem.o | test_irq_hdlr: test_irq_handler.o libserial.o libipc.o session.o usyscall.o libmem.o | ||||||
| 	@${ld} ${user_ldflags} -e main -o $@ $^ ${board_specs} | 	@${ld} ${user_ldflags} -e main -o $@ $^ ${board_specs} | ||||||
| 	@${objdump} -S $@ > $@.asm | 	@${objdump} -S $@ > $@.asm | ||||||
|  |  | ||||||
|  | @ -0,0 +1,20 @@ | ||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2020 AIIT XUOS Lab | ||||||
|  |  * XiUOS is licensed under Mulan PSL v2. | ||||||
|  |  * You can use this software according to the terms and conditions of the Mulan PSL v2. | ||||||
|  |  * You may obtain a copy of Mulan PSL v2 at: | ||||||
|  |  *        http://license.coscl.org.cn/MulanPSL2
 | ||||||
|  |  * THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, | ||||||
|  |  * EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, | ||||||
|  |  * MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. | ||||||
|  |  * See the Mulan PSL v2 for more details. | ||||||
|  |  */ | ||||||
|  | #include "libipc.h" | ||||||
|  | #include "libserial.h" | ||||||
|  | #include "usyscall.h" | ||||||
|  | 
 | ||||||
|  | IPC_SERVICES(IpcSwIntrHandler, Ipc_intr_3, Ipc_wait_intr_3); | ||||||
|  | 
 | ||||||
|  | enum { | ||||||
|  |     SW_INTERRUPT_3 = 3, | ||||||
|  | }; | ||||||
|  | @ -0,0 +1,35 @@ | ||||||
|  | /*
 | ||||||
|  |  * Copyright (c) 2020 AIIT XUOS Lab | ||||||
|  |  * XiUOS is licensed under Mulan PSL v2. | ||||||
|  |  * You can use this software according to the terms and conditions of the Mulan PSL v2. | ||||||
|  |  * You may obtain a copy of Mulan PSL v2 at: | ||||||
|  |  *        http://license.coscl.org.cn/MulanPSL2
 | ||||||
|  |  * THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, | ||||||
|  |  * EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, | ||||||
|  |  * MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. | ||||||
|  |  * See the Mulan PSL v2 for more details. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include "test_irq.h" | ||||||
|  | 
 | ||||||
|  | IPC_INTERFACE(Ipc_wait_intr_3, 1, ignore, 0); | ||||||
|  | int wait_intr(struct Session* session, void* ignore_param) | ||||||
|  | { | ||||||
|  |     return IPC_CALL(Ipc_wait_intr_3)(session, NULL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static char prog_name[] = "TEST_IRQ_BLOCK"; | ||||||
|  | int main(int argc, char* argv[]) | ||||||
|  | { | ||||||
|  |     struct Session session; | ||||||
|  |     if (connect_session(&session, "TestIRQ", 4096) < 0) { | ||||||
|  |         printf("connect session failed\n"); | ||||||
|  |         exit(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     printf("%s start waiting for IRQ.\n", prog_name); | ||||||
|  |     wait_intr(&session, NULL); | ||||||
|  |     printf("%s return from waiting for IRQ.\n", prog_name); | ||||||
|  | 
 | ||||||
|  |     exit(); | ||||||
|  | } | ||||||
|  | @ -10,35 +10,45 @@ | ||||||
|  * See the Mulan PSL v2 for more details. |  * See the Mulan PSL v2 for more details. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| #include "libipc.h" | #include "test_irq.h" | ||||||
| #include "libserial.h" |  | ||||||
| #include "usyscall.h" |  | ||||||
| 
 | 
 | ||||||
| IPC_SERVICES(IpcSwIntrHandler, Ipc_intr_3); | static bool has_one_interrupt = false; | ||||||
| 
 | int IPC_DO_SERVE_FUNC(Ipc_intr_3)(void* ignore) | ||||||
| enum { |  | ||||||
|     SW_INTERRUPT_3 = 3, |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| void sgi_test_handler(void) |  | ||||||
| { | { | ||||||
|     printf("TEST_SW_HDLR: In %s()\n", __func__); |     printf("TEST_SW_HDLR: In %s()\n", __func__); | ||||||
|  |     has_one_interrupt = true; | ||||||
|  |     return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int IPC_DO_SERVE_FUNC(Ipc_intr_3)(void* useless) | int IPC_DO_SERVE_FUNC(Ipc_wait_intr_3)(void* ignore) | ||||||
| { | { | ||||||
|     sgi_test_handler(); |     // delay the this handle
 | ||||||
|  |     if (!has_one_interrupt) { | ||||||
|  |         delay_session(); | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // serve can be done by now
 | ||||||
|  |     has_one_interrupt = false; | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| IPC_SERVER_INTERFACE(Ipc_intr_3, 1); | IPC_SERVER_INTERFACE(Ipc_intr_3, 1); | ||||||
| IPC_SERVER_REGISTER_INTERFACES(IpcSwIntrHandler, 1, Ipc_intr_3); | IPC_SERVER_INTERFACE(Ipc_wait_intr_3, 1); | ||||||
|  | IPC_SERVER_REGISTER_INTERFACES(IpcSwIntrHandler, 2, Ipc_intr_3, Ipc_wait_intr_3); | ||||||
| int main() | int main() | ||||||
| { | { | ||||||
|     if (register_irq(SW_INTERRUPT_3, Ipc_intr_3) == -1) { |     if (register_irq(SW_INTERRUPT_3, Ipc_intr_3) < 0) { | ||||||
|         printf("TEST_SW_HDLR: bind failed"); |         printf("TEST_SW_HDLR: bind failed"); | ||||||
|         exit(); |         exit(); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     static char prog_name[] = "TestIRQ"; | ||||||
|  |     if (register_server("TestIRQ") < 0) { | ||||||
|  |         printf("register server name: %s failed.\n", prog_name); | ||||||
|  |         exit(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     ipc_server_loop(&IpcSwIntrHandler); |     ipc_server_loop(&IpcSwIntrHandler); | ||||||
| 
 | 
 | ||||||
|     exit(); |     exit(); | ||||||
|  |  | ||||||
|  | @ -88,7 +88,7 @@ struct Inode { | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // directory entry
 | // directory entry
 | ||||||
| #define DIR_NAME_SIZE 14 | #define DIR_NAME_SIZE 30 | ||||||
| struct DirectEntry { | struct DirectEntry { | ||||||
|     uint16_t inum; |     uint16_t inum; | ||||||
|     char name[DIR_NAME_SIZE]; |     char name[DIR_NAME_SIZE]; | ||||||
|  |  | ||||||
|  | @ -156,6 +156,11 @@ void delay_session(void) | ||||||
|     session_delayed = true; |     session_delayed = true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | bool is_cur_session_delayed(void) | ||||||
|  | { | ||||||
|  |     return session_delayed; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void ipc_server_loop(struct IpcNode* ipc_node) | void ipc_server_loop(struct IpcNode* ipc_node) | ||||||
| { | { | ||||||
|     struct Session session_list[NR_MAX_SESSION]; |     struct Session session_list[NR_MAX_SESSION]; | ||||||
|  | @ -168,43 +173,46 @@ void ipc_server_loop(struct IpcNode* ipc_node) | ||||||
|         */ |         */ | ||||||
|         poll_session(session_list, NR_MAX_SESSION); |         poll_session(session_list, NR_MAX_SESSION); | ||||||
|         /* handle each session */ |         /* handle each session */ | ||||||
|         for (int i = 0; i < NR_MAX_SESSION; i++) { |         for (int repeat = 0; repeat <= 1; repeat++) { | ||||||
|             if (session_list[i].buf == NULL) { |             for (int i = 0; i < NR_MAX_SESSION; i++) { | ||||||
|                 yield(SYS_TASK_YIELD_NO_REASON); |                 if (session_list[i].buf == NULL) { | ||||||
|                 break; |                     yield(SYS_TASK_YIELD_NO_REASON); | ||||||
|             } |                     continue; | ||||||
|             cur_sess_id = session_list[i].id; |  | ||||||
|             struct IpcMsg* msg = IPCSESSION_MSG(&session_list[i]); |  | ||||||
|             /* handle every message in current session
 |  | ||||||
|                 a session could be delay in case one of its message(current message) needs to wait for an interrupt message's arrival |  | ||||||
|                 interfaces[opcode] should explicitly call delay_session() and return to delay this session |  | ||||||
|             */ |  | ||||||
|             while (msg->header.magic == IPC_MSG_MAGIC && msg->header.valid == 1 && msg->header.done == 0) { |  | ||||||
|                 // printf("session %d [%d, %d]\n", session_list[i].id, session_list[i].head, session_list[i].tail);
 |  | ||||||
|                 if (session_used_size(&session_list[i]) == 0 && session_forward_tail(&session_list[i], msg->header.len) < 0) { |  | ||||||
|                     break; |  | ||||||
|                 } |                 } | ||||||
| 
 |                 cur_sess_id = session_list[i].id; | ||||||
|                 // this is a message needs to handle
 |                 struct IpcMsg* msg = IPCSESSION_MSG(&session_list[i]); | ||||||
|                 if (ipc_node->interfaces[msg->header.opcode]) { |                 /* handle every message in current session
 | ||||||
|                     ipc_node->interfaces[msg->header.opcode](msg); |                     a session could be delay in case one of its message(current message) needs to wait for an interrupt message's arrival | ||||||
|                     // check if this session is delayed by op handler, all messages after the delayed message in current session is blocked.
 |                     interfaces[opcode] should explicitly call delay_session() and return to delay this session | ||||||
|                     if (session_delayed) { |                 */ | ||||||
|                         session_delayed = false; |                 while (msg->header.magic == IPC_MSG_MAGIC && msg->header.valid == 1 && msg->header.done == 0) { | ||||||
|  |                     // printf("session %d [%d, %d]\n", session_list[i].id, session_list[i].head, session_list[i].tail);
 | ||||||
|  |                     if (session_used_size(&session_list[i]) == 0 && session_forward_tail(&session_list[i], msg->header.len) < 0) { | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
|                 } else { | 
 | ||||||
|                     printf("Unsupport opcode(%d) for server: %s\n", msg->header.opcode, ipc_node->name); |                     // this is a message needs to handle
 | ||||||
|  |                     if (ipc_node->interfaces[msg->header.opcode]) { | ||||||
|  |                         ipc_node->interfaces[msg->header.opcode](msg); | ||||||
|  |                         // check if this session is delayed by op handler, all messages after the delayed message in current session is blocked.
 | ||||||
|  |                         if (is_cur_session_delayed()) { | ||||||
|  |                             msg->header.delayed = 1; | ||||||
|  |                             session_delayed = false; | ||||||
|  |                             break; | ||||||
|  |                         } | ||||||
|  |                     } else { | ||||||
|  |                         printf("Unsupport opcode(%d) for server: %s\n", msg->header.opcode, ipc_node->name); | ||||||
|  |                     } | ||||||
|  |                     // current msg is a message that needs to ignore
 | ||||||
|  |                     // finish this message in server's perspective
 | ||||||
|  |                     if (session_forward_head(&session_list[i], msg->header.len) < 0) { | ||||||
|  |                         break; | ||||||
|  |                     } | ||||||
|  |                     msg = IPCSESSION_MSG(&session_list[i]); | ||||||
|                 } |                 } | ||||||
|                 // current msg is a message that needs to ignore
 |                 // stop handle this session
 | ||||||
|                 // finish this message in server's perspective
 |                 cur_sess_id = -1; | ||||||
|                 if (session_forward_head(&session_list[i], msg->header.len) < 0) { |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
|                 msg = IPCSESSION_MSG(&session_list[i]); |  | ||||||
|             } |             } | ||||||
|             // stop handle this session
 |  | ||||||
|             cur_sess_id = -1; |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | @ -47,7 +47,7 @@ typedef struct { | ||||||
|             uint64_t valid : 1; // for server to peek new msg
 |             uint64_t valid : 1; // for server to peek new msg
 | ||||||
|             uint64_t done : 1; // for client to check request done
 |             uint64_t done : 1; // for client to check request done
 | ||||||
|             uint64_t init : 1; // for client to check request done
 |             uint64_t init : 1; // for client to check request done
 | ||||||
|             uint64_t reserved : 1; |             uint64_t delayed : 1; | ||||||
|             uint64_t nr_args : 4; |             uint64_t nr_args : 4; | ||||||
|             uint64_t opcode : 8; |             uint64_t opcode : 8; | ||||||
|             uint64_t len : 16; |             uint64_t len : 16; | ||||||
|  | @ -225,6 +225,7 @@ void ipc_server_loop(struct IpcNode* ipc_node); | ||||||
|         return res;                                                                                                                            \ |         return res;                                                                                                                            \ | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | bool is_cur_session_delayed(void); | ||||||
| #define IPC_SERVER_INTERFACE(ipc_name, argc)            \ | #define IPC_SERVER_INTERFACE(ipc_name, argc)            \ | ||||||
|     static int IPC_SERVE(ipc_name)(struct IpcMsg * msg) \ |     static int IPC_SERVE(ipc_name)(struct IpcMsg * msg) \ | ||||||
|     {                                                   \ |     {                                                   \ | ||||||
|  | @ -233,8 +234,10 @@ void ipc_server_loop(struct IpcNode* ipc_node); | ||||||
|             argv[i] = ipc_msg_get_nth_arg_buf(msg, i);  \ |             argv[i] = ipc_msg_get_nth_arg_buf(msg, i);  \ | ||||||
|         }                                               \ |         }                                               \ | ||||||
|         int32_t _ret = IPC_DO_SERVE##argc(ipc_name);    \ |         int32_t _ret = IPC_DO_SERVE##argc(ipc_name);    \ | ||||||
|         ipc_msg_set_return(msg, &_ret);                 \ |         if (!is_cur_session_delayed()) {                \ | ||||||
|         msg->header.done = 1;                           \ |             ipc_msg_set_return(msg, &_ret);             \ | ||||||
|  |             msg->header.done = 1;                       \ | ||||||
|  |         }                                               \ | ||||||
|         return 0;                                       \ |         return 0;                                       \ | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -80,7 +80,7 @@ struct Inode { | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // Directory is a file containing a sequence of DirEntry structures.
 | // Directory is a file containing a sequence of DirEntry structures.
 | ||||||
| #define DIR_NAME_SIZE 14 | #define DIR_NAME_SIZE 30 | ||||||
| struct DirEntry { | struct DirEntry { | ||||||
|     ushort inum; |     ushort inum; | ||||||
|     char name[DIR_NAME_SIZE]; |     char name[DIR_NAME_SIZE]; | ||||||
|  |  | ||||||
|  | @ -41,7 +41,7 @@ typedef struct { | ||||||
|             uint64_t valid : 1; // for server to peek new msg
 |             uint64_t valid : 1; // for server to peek new msg
 | ||||||
|             uint64_t done : 1; // for client to check request done
 |             uint64_t done : 1; // for client to check request done
 | ||||||
|             uint64_t init : 1; // for client to check request done
 |             uint64_t init : 1; // for client to check request done
 | ||||||
|             uint64_t reserved : 1; |             uint64_t delayed : 1; | ||||||
|             uint64_t nr_args : 4; |             uint64_t nr_args : 4; | ||||||
|             uint64_t opcode : 8; |             uint64_t opcode : 8; | ||||||
|             uint64_t len : 16; |             uint64_t len : 16; | ||||||
|  |  | ||||||
|  | @ -28,11 +28,20 @@ Modification: | ||||||
| 1. first version | 1. first version | ||||||
| *************************************************/ | *************************************************/ | ||||||
| #include "assert.h" | #include "assert.h" | ||||||
|  | #include "ipc.h" | ||||||
| #include "multicores.h" | #include "multicores.h" | ||||||
| #include "share_page.h" | #include "share_page.h" | ||||||
| #include "syscall.h" | #include "syscall.h" | ||||||
| #include "task.h" | #include "task.h" | ||||||
| 
 | 
 | ||||||
|  | #define IPCSESSION_MSG(session) ((struct IpcMsg*)((char*)((session)->buf) + (session)->head)) | ||||||
|  | 
 | ||||||
|  | static inline bool is_msg_needed(struct IpcMsg* msg) | ||||||
|  | { | ||||||
|  |     assert(msg != NULL); | ||||||
|  |     return msg->header.magic == IPC_MSG_MAGIC && msg->header.valid == 1 && msg->header.done == 0 && msg->header.delayed == 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| int sys_poll_session(struct Session* userland_session_arr, int arr_capacity) | int sys_poll_session(struct Session* userland_session_arr, int arr_capacity) | ||||||
| { | { | ||||||
|     struct TaskMicroDescriptor* cur_task = cur_cpu()->task; |     struct TaskMicroDescriptor* cur_task = cur_cpu()->task; | ||||||
|  | @ -71,10 +80,11 @@ int sys_poll_session(struct Session* userland_session_arr, int arr_capacity) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /* poll with new sessions */ |     /* poll with new sessions */ | ||||||
|     int i = 0; |     int nr_sessions_need_to_handle = 0; | ||||||
|  |     int session_idx = 0; | ||||||
|     DOUBLE_LIST_FOR_EACH_ENTRY(server_session, &cur_task->svr_sess_listhead, node) |     DOUBLE_LIST_FOR_EACH_ENTRY(server_session, &cur_task->svr_sess_listhead, node) | ||||||
|     { |     { | ||||||
|         if (i >= arr_capacity) { |         if (session_idx >= arr_capacity) { | ||||||
|             break; |             break; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -90,16 +100,24 @@ int sys_poll_session(struct Session* userland_session_arr, int arr_capacity) | ||||||
|             break; |             break; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         userland_session_arr[i++] = (struct Session) { |         userland_session_arr[session_idx] = (struct Session) { | ||||||
|             .buf = (void*)server_session->buf_addr, |             .buf = (void*)server_session->buf_addr, | ||||||
|             .capacity = server_session->capacity, |             .capacity = server_session->capacity, | ||||||
|             .head = server_session->head, |             .head = server_session->head, | ||||||
|             .tail = server_session->tail, |             .tail = server_session->tail, | ||||||
|             .id = SERVER_SESSION_BACKEND(server_session)->session_id, |             .id = SERVER_SESSION_BACKEND(server_session)->session_id, | ||||||
|         }; |         }; | ||||||
|  | 
 | ||||||
|  |         struct IpcMsg* msg = IPCSESSION_MSG(&userland_session_arr[session_idx]); | ||||||
|  |         if (is_msg_needed(msg)) { | ||||||
|  |             nr_sessions_need_to_handle++; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         session_idx++; | ||||||
|     } |     } | ||||||
|     if (LIKELY(i < arr_capacity)) { |     if (session_idx < arr_capacity && nr_sessions_need_to_handle == 0) { | ||||||
|         userland_session_arr[i].buf = 0; |         userland_session_arr[session_idx].buf = 0; | ||||||
|  |         sys_yield(SYS_TASK_YIELD_BLOCK_IPC); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return 0; |     return 0; | ||||||
|  |  | ||||||
|  | @ -76,6 +76,10 @@ static void send_irq_to_user(int irq_num) | ||||||
|         buf->header.magic = IPC_MSG_MAGIC; |         buf->header.magic = IPC_MSG_MAGIC; | ||||||
|         buf->header.valid = 1; |         buf->header.valid = 1; | ||||||
| 
 | 
 | ||||||
|  |         if (irq_forward_table[irq_num].handle_task->state == BLOCKED) { | ||||||
|  |             xizi_task_manager.task_unblock(irq_forward_table[irq_num].handle_task); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         /* add session head */ |         /* add session head */ | ||||||
|         session->head = (session->head + len) % session->capacity; |         session->head = (session->head + len) % session->capacity; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -31,7 +31,7 @@ Modification: | ||||||
| #include "syscall.h" | #include "syscall.h" | ||||||
| #include "task.h" | #include "task.h" | ||||||
| 
 | 
 | ||||||
| #include "log.h" | #include "assert.h" | ||||||
| 
 | 
 | ||||||
| int sys_yield(task_yield_reason reason) | int sys_yield(task_yield_reason reason) | ||||||
| { | { | ||||||
|  | @ -46,5 +46,17 @@ int sys_yield(task_yield_reason reason) | ||||||
|             xizi_task_manager.task_block(cur_task); |             xizi_task_manager.task_block(cur_task); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     // wake up all possible server
 | ||||||
|  |     struct client_session* client_session = NULL; | ||||||
|  |     DOUBLE_LIST_FOR_EACH_ENTRY(client_session, &cur_task->cli_sess_listhead, node) | ||||||
|  |     { | ||||||
|  |         assert(client_session != NULL); | ||||||
|  |         struct session_backend* session_backend = CLIENT_SESSION_BACKEND(client_session); | ||||||
|  |         if (session_backend->server->state == BLOCKED) { | ||||||
|  |             xizi_task_manager.task_unblock(session_backend->server); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     return 0; |     return 0; | ||||||
| } | } | ||||||
|  | @ -291,6 +291,7 @@ static void _task_yield_noschedule(struct TaskMicroDescriptor* task, bool blocki | ||||||
|         task->state = READY; |         task->state = READY; | ||||||
|     } |     } | ||||||
|     task->remain_tick = TASK_CLOCK_TICK; |     task->remain_tick = TASK_CLOCK_TICK; | ||||||
|  |     cur_cpu()->task = NULL; | ||||||
|     task_node_add_to_ready_list_back(task); |     task_node_add_to_ready_list_back(task); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue