1 /*- 2 * Copyright (c) 2010 Isilon Systems, Inc. 3 * Copyright (c) 2010 iX Systems, Inc. 4 * Copyright (c) 2010 Panasas, Inc. 5 * Copyright (c) 2013-2017 Mellanox Technologies, Ltd. 6 * All rights reserved. 7 * 8 * Redistribution and use in source and binary forms, with or without 9 * modification, are permitted provided that the following conditions 10 * are met: 11 * 1. Redistributions of source code must retain the above copyright 12 * notice unmodified, this list of conditions, and the following 13 * disclaimer. 14 * 2. Redistributions in binary form must reproduce the above copyright 15 * notice, this list of conditions and the following disclaimer in the 16 * documentation and/or other materials provided with the distribution. 17 * 18 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 19 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 20 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 21 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 22 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 23 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 24 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 25 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 26 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 27 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28 * 29 * $FreeBSD$ 30 */ 31 #ifndef _LINUXKPI_LINUX_WORKQUEUE_H_ 32 #define _LINUXKPI_LINUX_WORKQUEUE_H_ 33 34 #include "pthread.h" 35 #include "los_task.h" 36 #include "semaphore.h" 37 #include "los_memory.h" 38 #include "los_event.h" 39 #include "linux/list.h" 40 #include "linux/timer.h" 41 42 #ifdef __cplusplus 43 #if __cplusplus 44 extern "C" { 45 #endif /* __cplusplus */ 46 #endif /* __cplusplus */ 47 48 49 struct lock_class_key { 50 }; 51 52 #ifdef WORKQUEUE_SUPPORT_PRIORITY 53 /** 54 * @ingroup los_task 55 * Define a usable work priority. 56 * 57 * Highest task priority. 58 */ 59 #define OS_WORK_PRIORITY_HIGHEST 0 60 61 /** 62 * @ingroup los_task 63 * Define a usable work priority. 64 * 65 * Lowest task priority. 66 */ 67 #define OS_WORK_PRIORITY_LOWEST 31 68 69 70 /** 71 * @ingroup los_task 72 * Define a usable work priority. 73 * 74 * Default task priority. 75 */ 76 #define OS_WORK_PRIORITY_DEFAULT OS_WORK_PRIORITY_LOWEST 77 #endif 78 79 /** 80 * @ingroup workqueue 81 * System default workqueue. 82 */ 83 extern struct workqueue_struct *g_pstSystemWq; 84 85 /** 86 * @ingroup workqueue 87 * Workqueue task control block structure. 88 */ 89 typedef struct LosTaskCB task_struct; 90 91 /** 92 * @ingroup workqueue 93 * Define atomic_long_t as a signed integer. 94 * 95 */ 96 typedef long atomic_long_t; 97 98 struct work_struct; 99 100 /** 101 * @ingroup workqueue 102 * Workqueue handling function. 103 */ 104 typedef void (*work_func_t)(struct work_struct *); 105 106 /** 107 * @ingroup workqueue 108 * Work structure. 109 * A work is a node in a workqueue. 110 */ 111 struct work_struct { 112 atomic_long_t data; /**< Input parameter of the work handling function. */ 113 struct list_head entry; /**< Pointer to a doubly linked list of a work. */ 114 work_func_t func; /**< Work handling function. */ 115 unsigned int work_status; /**< Work status. */ 116 #ifdef WORKQUEUE_SUPPORT_PRIORITY 117 unsigned int work_pri; 118 #endif 119 }; 120 121 /** 122 * @ingroup workqueue 123 * delayed_work structure. 124 * A delayed_work is a work that is delayed to be mounted to a workqueue. 125 */ 126 struct delayed_work { 127 struct work_struct work; /**< Work structure. */ 128 struct timer_list timer; /**< Delay control block parameter structure. */ 129 struct workqueue_struct *wq; /**< Workqueue that contains the delayed_work structure. */ 130 int cpu; /**< Number of CPUs. Not in use temporarily. */ 131 }; 132 133 /** 134 * @ingroup workqueue 135 * Workqueue control structure. 136 * It awakes a workqueue or makes a workqueue sleep. 137 */ 138 typedef struct tag_cpu_workqueue_struct { 139 struct list_head worklist; /**< Pointer to a work doubly linked list. */ 140 struct work_struct *current_work; /**< Work that is being executed. */ 141 struct workqueue_struct *wq; /**< Workqueue that contains the workqueue control structure. */ 142 task_struct *thread; /**< Workqueue handling thread. */ 143 } cpu_workqueue_struct; 144 145 /** 146 * @ingroup workqueue 147 * Definition of a workqueue structure. 148 */ 149 struct workqueue_struct { 150 cpu_workqueue_struct *cpu_wq; /**< Workqueue control structure. */ 151 struct list_head list; /**< Pointer to a workqueue doubly linked list. */ 152 EVENT_CB_S wq_event; /**< Event of a workqueue. */ 153 unsigned int wq_id; /**< Workqueue ID. */ 154 int delayed_work_count; /**< Number of delayed works in a workqueue. */ 155 char *name; /**< Workqueue name. */ 156 int singlethread; /**< Whether to create a new working task. 0 indicates that 157 the default working task will be used. */ 158 int wq_status; /**< Workqueue status. */ 159 int freezeable; /**< Not in use temporarily. */ 160 int rt; /**< Not in use temporarily. */ 161 }; 162 163 /** 164 * @ingroup workqueue 165 * Work status enumeration. 166 */ 167 enum work_status { 168 WORK_BUSY_PENDING = 1U << 0, /**< The status of work item is pending execution. */ 169 WORK_BUSY_RUNNING = 1U << 1, /**< The status of work item is running. */ 170 WORK_STRUCT_PENDING = 1U << 0, /**< Work item is pending execution. */ 171 WORK_STRUCT_RUNNING = 1U << 1, /**< Work item is running. */ 172 }; 173 174 /** 175 * @ingroup workqueue 176 * Initialize a work. 177 */ 178 #ifdef WORKQUEUE_SUPPORT_PRIORITY 179 #define INIT_WORK(work, callbackFunc) do { \ 180 INIT_LIST_HEAD(&((work)->entry)); \ 181 (work)->func = (callbackFunc); \ 182 (work)->data = (atomic_long_t)(0); \ 183 (work)->work_status = 0; \ 184 (work)->work_pri = OS_WORK_PRIORITY_DEFAULT; \ 185 } while (0) 186 #else 187 #define INIT_WORK(work, callbackFunc) do { \ 188 INIT_LIST_HEAD(&((work)->entry)); \ 189 (work)->func = (callbackFunc); \ 190 (work)->data = (atomic_long_t)(0); \ 191 (work)->work_status = 0; \ 192 } while (0) 193 #endif 194 195 /** 196 * @ingroup workqueue 197 * Initialize a delayed work. 198 */ 199 #define INIT_DELAYED_WORK(work, func) \ 200 linux_init_delayed_work(work, func) 201 202 #define create_singlethread_workqueue(name) \ 203 linux_create_singlethread_workqueue(name) 204 205 #define create_workqueue(name) \ 206 linux_create_singlethread_workqueue(name) 207 208 #define destroy_workqueue(wq) \ 209 linux_destroy_workqueue(wq) 210 211 #define queue_work(wq, work) \ 212 linux_queue_work(wq, work) 213 214 #define queue_delayed_work(wq, dwork, delayTime) \ 215 linux_queue_delayed_work(wq, dwork, delayTime) 216 217 #define schedule_work(work) \ 218 linux_schedule_work(work) 219 220 #define schedule_delayed_work(dwork, delayTime) \ 221 linux_schedule_delayed_work(dwork, delayTime) 222 223 #define work_busy(work) \ 224 linux_work_busy(work) 225 226 #define flush_delayed_work(dwork) \ 227 linux_flush_delayed_work(dwork) 228 229 #define cancel_delayed_work(dwork) \ 230 linux_cancel_delayed_work(dwork) 231 232 #define cancel_delayed_work_sync(dwork) \ 233 linux_cancel_delayed_work_sync(dwork) 234 235 #define flush_work(work) \ 236 linux_flush_work(work) 237 238 #define cancel_work_sync(work) \ 239 linux_cancel_work_sync(work) 240 241 /* prototypes */ 242 243 /** 244 * @ingroup workqueue 245 * @brief Initialize a delayed work. 246 * 247 * @par Description: 248 * This API is used to initialize a delayed work. 249 * 250 * @attention 251 * <ul> 252 * <li>The parameter dwork and func shoud be valid memory, otherwise, the system may be abnormal. </li> 253 * </ul> 254 * 255 * @param dwork [IN] Work handle. 256 * @param func [IN] Executive function. 257 * 258 * @retval None. 259 * @par Dependency: 260 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 261 * @see none. 262 */ 263 extern void linux_init_delayed_work(struct delayed_work *dwork, work_func_t func); 264 265 /** 266 * @ingroup workqueue 267 * @brief Create a workqueue. 268 * 269 * @par Description: 270 * This API is used to create a workqueue that has a specified name. 271 * 272 * @attention 273 * <ul> 274 * <li> The passed-in workqueue name is a character string that cannot be null 275 * and is the only identifier of the workqueue, make sure it is unique. </li> 276 * </ul> 277 * 278 * @param name [IN] Workqueue name. 279 * 280 * @retval NULL The workqueue fails to be created. 281 * @retval workqueue_struct* The workqueue is successfully created. 282 * @par Dependency: 283 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 284 * @see destroy_workqueue 285 */ 286 extern struct workqueue_struct *linux_create_singlethread_workqueue(char *name); 287 288 /** 289 * @ingroup workqueue 290 * @brief Delete a workqueue. 291 * 292 * @par Description: 293 * This API is used to delete a workqueue that has a specified handle. 294 * 295 * @attention 296 * <ul> 297 * <li>The name of the workqueue will be null and the workqueue cannot be used again 298 * after the workqueue is deleted. </li> 299 * </ul> 300 * 301 * @param wq [IN] Workqueue handle. 302 * 303 * @retval None. 304 * @par Dependency: 305 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 306 * @see create_workqueue 307 */ 308 extern void linux_destroy_workqueue(struct workqueue_struct *wq); 309 310 /** 311 * @ingroup workqueue 312 * @brief Queue a work on a workqueue. 313 * 314 * @par Description: 315 * This API is used to queue a work on a specified workqueue. 316 * 317 * @attention 318 * <ul> 319 * <li>The parameter wq and work shoud be valid memory, otherwise, the system may be abnormal. </li> 320 * <li>The work will be immediately queued on the workqueue. </li> 321 * </ul> 322 * 323 * @param wq [IN] Workqueue handle. 324 * @param work [IN] Work handle. 325 * 326 * @retval #TRUE The work is successfully queued on the workqueue. 327 * @retval #FALSE The work fails to be queued on the workqueue. 328 * @par Dependency: 329 * <ul> 330 * <li>This function should be used after create_singlethread_workqueue() or create_workqueue() has been called</li> 331 * <li>workqueue.h: the header file that contains the API declaration.</li> 332 * </ul> 333 * @see cancel_work_sync 334 */ 335 extern bool linux_queue_work(struct workqueue_struct *wq, struct work_struct *work); 336 337 /** 338 * @ingroup workqueue 339 * @brief Queue a work on a workqueue after delay. 340 * 341 * @par Description: 342 * This API is used to queue a work on a specified workqueue after delay. 343 * 344 * @attention 345 * <ul> 346 * <li>The parameter wq and dwork shoud be valid memory, otherwise, the system may be abnormal. </li> 347 * <li>The work will be queued on the workqueue in a delayed period of time. </li> 348 * <li>The work will be queued on the workqueue immediately if delayTime is 0, it as same as queue_work(). </li> 349 * </ul> 350 * 351 * @param wq [IN] Workqueue handle. 352 * @param dwork [IN] Delayed work handle. 353 * @param delayTime [IN] Delayed time, number of ticks to wait or 0 for immediate execution. 354 * 355 * @retval #TRUE The work is successfully queued on the workqueue. 356 * @retval #FALSE The work fails to be queued on the workqueue. 357 * @par Dependency: 358 * <ul> 359 * <li>This function should be used after create_singlethread_workqueue() or create_workqueue() has been called</li> 360 * <li>workqueue.h: the header file that contains the API declaration.</li> 361 * </ul> 362 * @see cancel_delayed_work 363 */ 364 extern bool linux_queue_delayed_work(struct workqueue_struct *wq, struct delayed_work *dwork, unsigned int delayTime); 365 366 /** 367 * @ingroup workqueue 368 * @brief Put a work in a default workqueue. 369 * 370 * @par Description: 371 * This API is used to put a work in the default workqueue that is created when Huawei LiteOS is initialized. 372 * 373 * @attention 374 * <ul> 375 * <li>The parameter work shoud be valid memory, otherwise, the system may be abnormal. </li> 376 * <li>The default workqueue is g_pstSystemWq. </li> 377 * </ul> 378 * 379 * @param work [IN] Work handle. 380 * 381 * @retval #TRUE The work is successfully put in the workqueue. 382 * @retval #FALSE The work fails to be put in the workqueue. 383 * @par Dependency: 384 * <ul> 385 * <li>This function should be used after create_singlethread_workqueue() or create_workqueue() has been called</li> 386 * <li>workqueue.h: the header file that contains the API declaration.</li> 387 * </ul> 388 * @see cancel_work_sync 389 */ 390 extern bool linux_schedule_work(struct work_struct *work); 391 392 /** 393 * @ingroup workqueue 394 * @brief Put a work in a default workqueue after delay. 395 * 396 * @par Description: 397 * This API is used to put a work in the default workqueue that is created 398 * when Huawei LiteOS is initialized in a delayed period of time. 399 * 400 * @attention 401 * <ul> 402 * <li>The parameter dwork shoud be valid memory, otherwise, the system may be abnormal. </li> 403 * <li>The default workqueue is g_pstSystemWq. </li> 404 * <li>The dwork will be queued on the workqueue immediately if delayTime is 0. </li> 405 * </ul> 406 * 407 * @param dwork [IN] Delayed work handle. 408 * @param delayTime [IN] Delayed time, number of ticks to wait or 0 for immediate execution. 409 * 410 * @retval #TRUE The work is successfully put in the workqueue. 411 * @retval #FALSE The work fails to be put in the workqueue. 412 * @par Dependency: 413 * <ul> 414 * <li>This function should be used after create_singlethread_workqueue() or create_workqueue() has been called</li> 415 * <li>workqueue.h: the header file that contains the API declaration.</li> 416 * </ul> 417 * @see cancel_delayed_work 418 */ 419 extern bool linux_schedule_delayed_work(struct delayed_work *dwork, unsigned int delayTime); 420 421 /** 422 * @ingroup workqueue 423 * @brief Query the work status. 424 * 425 * @par Description: 426 * This API is used to query the status of a work and a delayed work. 427 * 428 * @attention 429 * <ul> 430 * <li>The parameter work shoud be valid memory, otherwise, the system may be abnormal.</li> 431 * </ul> 432 * 433 * @param work [IN] Work handle. 434 * 435 * @retval #WORK_BUSY_PENDING The work is pending. 436 * @retval #WORK_BUSY_RUNNING The work is running. 437 * @retval #FALSE The value of the parameter work is NULL. 438 * @par Dependency: 439 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 440 * @see None. 441 */ 442 extern unsigned int linux_work_busy(struct work_struct *work); 443 444 /** 445 * @ingroup workqueue 446 * @brief Immediately execute a delayed work. 447 * 448 * @par Description: 449 * This API is used to immediately put a delayed work in a workqueue 450 * and wait for the execution of the delayed work to end. 451 * 452 * @attention 453 * <ul> 454 * <li>flush_delayed_work() should be used after queue_delayed_work() has been called.</li> 455 * <li>The parameter dwork shoud be valid memory, otherwise, the system may be abnormal.</li> 456 * </ul> 457 * 458 * @param dwork [IN] Delayed work handle. 459 * 460 * @retval #TRUE The operation succeeds. 461 * @retval #FALSE The operation fails. 462 * @par Dependency 463 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 464 * @see None. 465 */ 466 extern bool linux_flush_delayed_work(struct delayed_work *dwork); 467 468 /** 469 * @ingroup workqueue 470 * @brief Cancel a delayed work. 471 * 472 * @par Description: 473 * This API is used to cancel a delayed work, which means that the work will not be executed regardless of 474 * whether the delayed time has expired. 475 * 476 * @attention 477 * <ul> 478 * <li>cancel_delayed_work() should be used after queue_delayed_work() has been called.</li> 479 * <li>The parameter dwork shoud be valid memory, otherwise, the system may be abnormal.</li> 480 * </ul> 481 * 482 * @param dwork [IN] Delayed work handle. 483 * 484 * @retval #TRUE The delayed work is successfully canceled. 485 * @retval #FALSE The delayed work fails to be canceled. 486 * @par Dependency: 487 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 488 * @see queue_delayed_work 489 */ 490 extern bool linux_cancel_delayed_work(struct delayed_work *dwork); 491 /** 492 * @ingroup workqueue 493 * @brief Cancel a delayed work and wait for it to finish. 494 * 495 * @par Description: 496 * This API is used to cancel a delayed work, which means that the work will not be executed regardless of 497 * whether the delayed time has expired. 498 * 499 * @attention 500 * <ul> 501 * <li>cancel_delayed_work_sync() should be used after queue_delayed_work() has been called.</li> 502 * <li>The parameter dwork shoud be valid memory, otherwise, the system may be abnormal.</li> 503 * </ul> 504 * 505 * @param dwork [IN] Delayed work handle. 506 * 507 * @retval #TRUE The delayed work is successfully canceled. 508 * @retval #FALSE The delayed work fails to be canceled. 509 * @par Dependency: 510 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 511 * @see queue_delayed_work 512 */ 513 extern bool linux_cancel_delayed_work_sync(struct delayed_work *dwork); 514 515 /** 516 * @ingroup workqueue 517 * @brief Immediately execute a work. 518 * 519 * @par Description: 520 * This API is used to immediately execute a specified work and wait for the execution to end. 521 * 522 * @attention 523 * <ul> 524 * <li>flush_work() should be used after queue_work() has been called.</li> 525 * <li>The parameter work shoud be valid memory, otherwise, the system may be abnormal.</li> 526 * </ul> 527 * 528 * @param work [IN] Work handle. 529 * 530 * @retval #TRUE The operation succeeds. 531 * @retval #FALSE The operation fails. 532 * @par Dependency: 533 * <ul><li>workqueue.h: the header file that contains the API declaration.</li></ul> 534 * @see None. 535 */ 536 extern bool linux_flush_work(struct work_struct *work); 537 538 /** 539 * @ingroup workqueue 540 * @brief Cancel a work. 541 * 542 * @par Description: 543 * This API is used to cancel a work that is pending or running. 544 * 545 * @attention 546 * <ul> 547 * <li>cancel_work_sync() should be used after queue_work() has been called.</li> 548 * <li>The parameter work shoud be valid memory, otherwise, the system may be abnormal.</li> 549 * </ul> 550 * 551 * @param work [IN] Work handle. 552 * 553 * @retval #TRUE The work is successfully canceled. 554 * @retval #FALSE The work fails to be canceled. 555 * @par Dependency: 556 * <ul><li>workqueue.h: the header file that contain the API declaration.</li></ul> 557 * @see queue_work 558 */ 559 extern bool linux_cancel_work_sync(struct work_struct *work); 560 561 #ifdef __cplusplus 562 #if __cplusplus 563 } 564 #endif /* __cplusplus */ 565 #endif /* __cplusplus */ 566 567 #endif /* _LINUXKPI_LINUX_WORKQUEUE_H_ */ 568