lvoid□□(\*irq eoi)(struct irq data \*data);

□unsigned long□flags;

Linux中断处理流程(基于Linux-5.10.4) 作者: 王利涛 视频配套地址: https://wanglitao.taobao.com 普通进程 vector\_stub宏定义 中断处理,CPU硬件自动完成的部分 macro□vector\_stub, name, mode, correction=0 系统启动过程中,每个控制器(irq\_domain)初始化,会建立硬件中断号(HW interrupt ID)和软中断号(IRQ number)之间的映射 存CPSR到SPSR irq寄存器 align□5 b置CPSR控制位, 让CPU进入ARM状态、IRQ模式 SR中的IRQ位置一,硬件自动关闭IRQ static int gic\_irq\_domain\_map(struct irq\_domain \*d, unsigned int irq, irq\_hw\_number\_t hw) vector\_\name: 当前中断地址(返回地址)保存到LR\_irq寄存器  $\Box$ . if \correction 设置PC指针PC=0x00000018,跳转到中断向量表执行 □sub□lr, lr, #\correction struct gic\_chip\_data \*gic = d->host\_data; struct irq\_data \*irqd = irq\_desc\_get\_irq\_data(irq\_to\_desc(irq)); □@ (parent CPSR) ARM 中断向量表 switch (hw) {  $\square$ stmia $\square$ sp, {r0, lr} $\square$ @ save r0, lr case 0 ... 15: □mrs□1r, spsr lirq set percpu devid(irq) section .vectors, "ax", %progbits □str□lr, [sp, #8]□@ save spsr irq\_domain\_set\_info(d, irq, hw, &gic->chip, d->host\_data, handle\_percpu\_devid\_fasteoi\_ipi, NULL, NULL); vectors start: W(b) □vector\_rst ]case 16 ... 31:  $\square$ W(b)  $\square$ vector\_und @ Prepare for SVC32 mode. IRQs remain disabled. □ irq\_set\_percpu\_devid(irq)  $\square$ W(1dr) $\square$ pc, .L\_vectors\_start + 0x1000 lirq\_domain\_set\_info(d, irq, hw, &gic->chip, d->host\_data, handle\_percpu\_devid\_irq, NULL, NULL); □W(b)□vector\_pabt □mrs□r0, cpsr  $\square$ W(b)  $\square$ vector\_dabt □eor□r0, r0, #(\mode^SVC\_MODE|PSR\_ISETSTATE)  $\square$ W(b)  $\square$ vector\_addrexcptn □msr□spsr\_cxsf, r0  $\exists irq\_domain\_set\_info(d, irq, hw, &gic->chip, d->host\_data, handle\_fasteoi\_irq, NULL, NULL);$ □W(b)□vector\_irq lirq set probe(irq);  $\square$ W(b)  $\square$ vector fiq  $\square$  and  $\square$  1r, 1r, #0x0f □irqd\_set\_single\_target(irqd)  $THUMB(\square adr \square r0, 1f \square \square)$ THUMB ( $\Box 1 dr \Box 1r$ , [r0, 1r, 1s1 #2]  $\Box$ ) irq usr □mov□r0, sp /\* Prevents SW retriggers which mess up the ACK/EOI ordering \*/  $ARM(\Box 1dr \Box 1r, [pc, 1r, 1s1 #2] \Box)$ irg usr: □movs□pc, 1r@ branch to handler in SVC mode lirqd\_set\_handle\_enforce\_irqctx(irqd); □usr\_entry ENDPROC(vector\_\name) □kuser\_cmpxchg\_check lirq\_handler IRQ 中断向量表 □get thread info tsk arch/arm/kernel/entry-armv.S □mov□why, #0 □b□ret\_to\_user\_from\_irq /\* Interrupt dispatcher \*/  $UNWIND(.fnend \square \square)$ vector\_stub irq, IRQ\_MODE, 4 ENDPROC(\_\_irq\_usr) long irq\_usr @ 0 (USR\_26 / USR\_32) □.long□\_irq\_invalid□@ 1 (FIQ\_26 / FIQ\_32) \_irq\_do\_set\_handler(struct irq\_desc \*desc, irq\_flow\_handler\_t □.long□\_irq\_invalid□@ 2 (IRQ 26 / IRQ 32) /\* Interrupt handling \*/ ndle, int is\_chained, const char \*name)  $\square$ . long  $\square$  irq svc  $\square$  @ 3 (SVC\_26 / SVC\_32)  $\square$ . macro $\square$  irg handler  $\square$ .long  $\square$ \_irq\_invalid  $\square$  @ 4 #ifdef CONFIG GENERIC IRQ MULTI HANDLER □.long□\_\_irq\_invalid□@ 5 □ldr□r1, =handle\_arch\_irq □desc->handle\_irq = handle; ← - - - - - $\square$ .long $\square$ \_irq\_invalid $\square$ @ 6  $\square$ mov $\square$ r0, sp  $\square desc- > name = name;$  $\square$ .long $\square$ \_irq\_invalid $\square$ @ 7 □badr□lr, 9997f ].long□\_\_irq\_invalid□@ 8  $\square$ 1dr $\square$ pc, [r1] J.long□\_\_irq\_invalid□@ 9 ].long□\_\_irq\_invalid□@ a □arch\_irq\_handler\_default  $\square$ .long $\square$ \_irq\_invalid $\square$ @ b ].long□\_\_irq\_invalid□@ c .long□\_\_irq\_invalid□@ d 根据HW interrupt ID找到IRQ number,调用asm\_do\_IRQ irqreturn\_t \_\_handle\_irq\_event\_percpu(struct irq\_desc \*desc, void handle\_fasteoi\_irq(struct irq\_desc \*desc) .long□\_\_irq\_invalid□@ e unsigned int \*flags) ].long□\_\_irq\_invalid□@ f Struct irq\_chip \*chip = desc->irq\_data.chip; /\* Interrupt handling. Preserves r7, r8, r9\*/ lirqreturn\_t retval = IRQ\_NONE; □raw\_spin\_lock(&desc->lock); lunsigned int irq = desc->irq\_data.irq; □desc->istate &= ~(IRQS\_REPLAY | IRQS\_WAITING) .macro□arch irq handler default ch/arm/kernel/irq.c lstruct irqaction \*action; □kstat\_incr\_irqs\_this\_cpu(desc); □get irqnr preamble r6, lr asmlinkage void \_\_exception\_irq\_entry if (desc->istate & IRQS ONESHOT) 1: □get\_irqnr\_and\_base r0, r2, r6, lr \_asm\_do\_IRQ(unsigned int irq, struct pt\_regs \*regs) lfor\_each\_action\_of\_desc(desc, action) {  $\square \square$ mask $_{
m irq}$ (desc);  $\square$ movne $\square$ r1, sp □irqreturn\_t res;  $\square$  handle\_irq\_event(desc); □handle\_IRQ(irq, regs);  $\square$  trace\_irq\_handler\_entry(irq, action); □ cond unmask eoi irq(desc, chip);  $\square$ @ routine called with r0 = irq number  $\Box$ res = action->handler(irq, action->dev\_id); □raw\_spin\_unlock(&desc->lock); □@, r1 = struct pt\_regs \*  $\Box$ trace\_irq\_handler\_exit(irq, action, res); □badrne□lr, 1b oid handle\_IRQ(unsigned int irq, struct pt\_regs \*regs) l□retval |= res; □bne□asm do IRQ if (!(chip->flags & IRQCHIP\_EOI IF HANDLED)) \_\_handle\_domain\_irq(NULL, irq, false, regs); □return retval; □chip->irq\_eoi(&desc->irq\_data);  $\Box$ raw\_spin\_unlock(&desc->lock); /\* Architectures call this to let the generic IRQ layer handle an interrupt\*/ static inline void generic\_handle\_irq\_desc(struct irq\_desc \*desc) \_\_handle\_domain\_irq(struct irq\_domain \*domain, unsigned int hwirq, □□bool lookup, struct pt\_regs \*regs) irgreturn\_t handle\_irg\_event(struct irg\_desc \*desc) desc=>handle irg(desc); lstruct pt\_regs \*old\_regs = set\_irq\_regs(regs); Jirqreturn\_t ret;  $\sqcup$  int ret = 0; □desc->istate &= ~IRQS\_PENDING; lirqd\_set(&desc->irq\_data, IRQD\_IRQ\_INPROGRESS);  $\square$  irq\_enter(); □raw spin unlock(&desc->lock); #ifdef CONFIG IRQ DOMAIN □ret = handle\_irq\_event\_percpu(desc); if (lookup)  $\square \square irq = irq_find_mapping(domain, hwirq);$ t request\_threaded\_irq(unsigned int irq, irq\_handler\_t handler, lraw\_spin\_lock(&desc->lock);  $\square$  if (unlikely(!irq || irq >= nr\_irqs)) { irqd\_clear(&desc->irq\_data, IRQD\_IRQ\_INPROGRESS); □irq\_handler\_t thread\_fn, unsigned long irqflags, □const char \*devname, void \*dev\_id)  $\square \square$  ack bad irg(irg); return ret;  $\square \square ret = -EINVAL;$ struct irgaction \*action;  $|\Box$ } else {  $\square \square$ generic\_handle\_irq(irq); □struct irq\_desc \*desc; ∃int retval; □irq\_exit(); ☐if (irq == IRQ\_NOTCONNECTED) □return -ENOTCONN;  $\square$  set\_irq\_regs(old\_regs); □return ret: if (((irgflags & IRQF SHARED) && !dev id) (!(irqflags & IRQF\_SHARED) && (irqflags & IRQF\_COND\_SUSPEND)) || ((irqflags & IRQF\_NO\_SUSPEND) && int generic\_handle\_irq(unsigned int irq) (irqflags & IRQF\_COND\_SUSPEND))) irq\_desc[NR\_IRQS] □return -EINVAL; \□struct irq\_desc \*desc = irq\_to\_desc(irq); □struct irq\_data \*data; truct irq\_desc { desc = irq\_to\_desc(irq); □struct irq\_common\_data□ irq\_common\_data; if (!desc) □data = irq\_desc\_get\_irq\_data(desc); lstruct irq\_data□□irq\_data; □□return -EINVAL; □if (WARN\_ON\_ONCE(!in\_irq() && handle\_enforce\_irqctx(data))) lunsigned int \_\_percpu□\*kstat\_irqs; □□return -EPERM; lirq\_flow\_handler\_t□handle\_irq; if (!irq\_settings\_can\_request(desc) | lstruct irqaction□\*action;□/\* IRQ action list \*/ WARN\_ON(irq\_settings\_is\_per\_cpu\_devid(desc))) □generic\_handle\_irq\_desc(desc); □return -EINVAL; □return 0;  $\square$ atomic\_t $\square$   $\square$ threads\_handled; if (\text{\text{Mandler}}) {  $\square$  if (!thread fn)□□return -EINVAL; truct irgaction { □□handler = irq\_default\_primary\_handler; static inline int \_\_must\_check  $lirq_handler_t \square \square handler;$ request\_irq(unsigned int irq, irq\_handler\_t handler, unsigned long  $\exists void \Box \Box \Box * dev_id;$ flags, const char \*name, void \*dev) □void \_\_percpu□□\*percpu\_dev\_id; action =\kzalloc(sizeof(struct irqaction), GFP\_KERNEL); □struct irqaction□\*next; if (!action) return request\_threaded\_irq(irq, handler, NULL, flags, name, dev);  $\exists irq\_handler\_t \square \square thread\_fn;$ **←** □□return -ENOMEM; □struct task\_struct□\*thread; ∃struct irqaction□\*secondary; □action->handler = handler; □action→thread\_fn = thread\_fn; □unsigned int□□irq; □unsigned int□□flags;  $\square$ action->flags = irqflags;  $\square$  wakeup\_softirqd();  $\square$ unsigned long $\square$   $\square$ thread\_flags;  $\square$ action->name = devname;  $\square$ unsigned long $\square$   $\square$ thread\_mask; □action->dev\_id = dev\_id; static int \_\_init rtc\_init(void)  $\square$  const char  $\square$   $\square$  \*name; □struct proc\_dir\_entry□\*dir; □retval = irq\_chip\_pm\_get(&desc->irq\_data);  $\square$  if (retval < 0) { irqreturn\_t ret = 0; □□kfree(action); truct irq\_data { □□return retval; regs = (rtc\_reg\_t \*)ioremap(RTC\_BASE, sizeof(rtc\_reg\_t)); lu32□□□mask; printk("rtc\_init\n"); / □unsigned int□□irq; □unsigned long□□hwirq; Iretval = \_\_setup\_irq(irq, desc, action); set\_rtc\_alarm(regs); / □struct irq\_common\_data□\*common; □struct irq\_chip□□\*chip; ∃if (retval) { ret = request\_irq(39, rtc\_alarm\_handler, 0, "rtc0-test", NULL); □struct irq\_domain□\*domain; Dirq\_chip\_pm\_put(&desc->irq\_data); if (ret == -1) { □void□□□\*chip\_data; □kfree(action->secondary); printk("request\_irq failed!\n"); □□kfree(action); return −1; truct irq\_chip { □struct device□\*parent\_device; #ifdef CONFIG DEBUG SHIRQ FIXME return 0; □const char□\*name; if (!retval && (irqflags & IRQF\_SHARED)) { □unsigned int□(\*irq\_startup)(struct irq\_data \*data); □unsigned long flags; lvoid□□(\*irq\_shutdown)(struct irq\_data \*data); void□□(\*irq\_enable)(struct irq\_data \*data); □□disable\_irq(irq); void□□(\*irq\_disable)(struct\_irq\_data \*data);  $\square \square local\_irq\_save(flags);$  $void \square \square$  (\*irq\_ack) (struct irq\_data \*data); □ handler(irq, dev\_id); lvoid□□(\*irq\_mask)(struct irq\_data \*data); □local\_irq\_restore(flags) lvoid□□(\*irq\_mask\_ack)(struct irq\_data \*data);  $\square$  enable\_irq(irq); lvoid□□(\*irq\_unmask)(struct\_irq\_data \*data);

□return retval;

tasklet的执行过程 el/softirq.c oid \_\_init softirq\_init(void) □int cpu; hile (list) { 软中断的执行过程 (ksoftirqd\_running(local\_softirq\_pending())) ef CONFIG\_HAVE\_IRQ\_EXIT\_ON\_IRQ\_STACK \_\_do\_softirq(); do softirg own stack();

Ifor\_each\_possible\_cpu(cpu) {  $\square$ per\_cpu(tasklet\_vec, cpu).tail = &per\_cpu(tasklet\_vec, cpu).head; □per\_cpu(tasklet\_hi\_vec, cpu).tail = &per\_cpu(tasklet\_hi\_vec, cpu).head; □open\_softirq(TASKLET\_SOFTIRQ, tasklet\_action); Jopen\_softirq(HI\_SOFTIRQ, tasklet\_hi\_action); static \_\_latent\_entropy void tasklet\_action(struct softirq\_action \*a)  $\square$  tasklet action common(a, this cpu ptr(&tasklet vec), TASKLET SOFTIRQ); id tasklet\_action\_common(struct softirq\_action \*a, struct tasklet\_head \*tl\_head, □□ unsigned int softirq\_nr) ]struct tasklet\_struct \*list; llocal\_irq\_disable();  $lst = t1 head \rightarrow head;$ \_head->head = NULL; 1\_head->tail = &tl\_head->head; local\_irq\_enable(); |struct tasklet\_struct \*t = list; list = list->next; if (tasklet\_trylock(t)) if (!atomic\_read(&t->count)) if (!test\_and\_clear\_bit(TASKLET\_STATE\_SCHED,&t->state)) if (t->use\_callback)  $\exists \Box t \rightarrow callback(t);$ d raise softirg(unsigned int nr) else  $\square \square t \rightarrow func(t \rightarrow data);$ nsigned long flags; □ tasklet\_unlock(t) local irq save(flags); continue; □raise softirg irgoff(nr); □local\_irq\_restore(flags); □tasklet\_unlock(t); llocal irg disable() id raise\_softirq\_irqoff(unsigned int nr) t->next = NULL; |\*tl\_head->tail = t; raise softirg irqoff(nr); tl\_head->tail = &t->next; (!in\_interrupt()) \_\_raise\_softirq\_irqoff(softirq\_nr);  $\square \square$  wakeup softirqd(); local\_irq\_enable(); \_\_raise\_softirq\_irqoff(unsigned int nr) ckdep\_assert\_irqs\_disabled() .ce\_softirq\_raise(nr); softirq pending(1UL << nr);</pre> d open\_softirq(int nr, void (\*action)(struct softirq\_action \*)) oftirq\_vec[nr].action = action; nlinkage \_\_visible void \_\_softirq\_entry \_\_do\_softirq(void) unsigned long end = jiffies + MAX\_SOFTIRQ\_TIME; unsigned long old\_flags = current->flags; int max\_restart = MAX\_SOFTIRQ\_RESTART; d irq\_exit(void) struct softirq\_action \*h; ool in\_hardirq; irq\_exit\_rcu(); \_u32 pending; u ira exit(): nt softirq\_bit; |lockdep\_hardirq\_exit(); current->flags &= ~PF\_MEMALLOC; id irq\_exit\_rcu(void) 🛛 🛹 pending = local\_softirq\_pending(); account\_irq\_enter\_time(current); \_local\_bh\_disable\_ip(\_RET\_IP\_, SOFTIRQ\_OFFSET); ockdep\_hardirq\_exit(); n\_hardirq = lockdep\_softirq\_start(); atic inline void \_\_irq\_exit\_rcu(void) 🎽 set\_softirq\_pending(0); fndef \_\_ARCH\_IRQ\_EXIT\_IRQS\_DISABLED .ocal\_irq\_enable(); local\_irq\_disable(); lockdep\_assert\_irqs\_disabled(); = softirq\_vec; account irg exit time(current); while ((softirq\_bit = ffs(pending))) { preempt\_count\_sub(HARDIRQ\_OFFSET); □unsigned int vec\_nr; (!in\_interrupt() && local\_softirq\_pending()) lint prev\_count;  $\square$  invoke\_softirq(); h += softirq\_bit - 1 ck\_irq\_exit(); /ec/nr = h - softirq\_vec; ]prev\_count = preempt\_count(); tic inline void invoke\_softirq(void) Zkstat\_incr\_softirgs\_this\_cpu(vec\_nr);

ltrace\_softirq\_entry(vec\_nr);

Itrace\_softirq\_exit(vec\_nr);

□pending >>= softirq\_bit;

cu\_softirq\_qs();

--max\_restart)

□□goto restart;

 $\square$  wakeup\_softirqd();

.ocal\_irq\_disable();

□pr\_err("huh, entered softirq %u %s %p with

prev\_count, preempt\_count());

(\_this\_cpu\_read(ksoftirqd) == current)

□preempt\_count\_set(prev\_count);

pending = local\_softirq\_pending();

.ockdep\_softirq\_end(in\_hardirq);

local bh enable (SOFTIRQ OFFSET);

current\_restore\_flags(old\_flags, PF\_MEMALLOC);

iccount\_irq\_exit\_time(current);

□WARN\_ON\_ONCE(in\_interrupt());

vec nr, softirg to name[vec nr], h->action,

if (time\_before(jiffies, end) && !need\_resched() &&

preempt\_count %08x, exited with %08x?\n",

]h∕->action(h);

workqueue工作队列工作流程

schedule work (struct work\_struct \*work) turn queue work(system wq, work); queue work (struct workqueue\_struct \*wq, struct work\_struct \*work) urn queue\_work\_on(WORK\_CPU\_UNBOUND, wq, work);

queue work on (int cpu, struct workqueue\_struct \*wq, struct work\_struct \*work) l ret = false; igned long flags; al\_irq\_save(flags); (!test and set bit(WORK STRUCT PENDING BIT, work data bits(work))) \_\_queue\_work(cpu, wq, work); ret = true; al\_irq\_restore(flags); urn ret; queue work (int cpu, struct workqueue\_struct \*wq, struct work\_struct \*work) ict pool\_workqueue \*pwq; ruct worker\_pool \*last\_pool; ict list\_head \*worklist; igned int work\_flags; igned int req\_cpu = cpu; ekdep\_assert\_irqs\_disabled() bug work activate(work); (unlikely(wq->flags & WQ DRAINING) && WARN\_ON\_ONCE(!is\_chained\_work(wq))) \_read\_lock(); (wq->flags & WQ\_UNBOUND) { f (reg cpu == WORK CPU UNBOUND)  $\exists cpu = wq select unbound cpu(raw smp processor id());$ lpwq = unbound\_pwq\_by\_node(wq, cpu\_to\_node(cpu)); f (req\_cpu == WORK CPU UNBOUND)  $\exists cpu = raw_smp_processor_id();$ lpwq = per\_cpu\_ptr(wq->cpu\_pwqs, cpu); st pool = get work pool(work); (last pool && last pool != pwq->pool) truct worker \*worker; raw\_spin\_lock(&last\_pool->lock); orker = find\_worker\_executing\_work(last\_pool, work); f (worker && worker->current\_pwq->wq == wq) { wq = worker->current\_pwq; Traw spin unlock(&last pool->lock); lraw spin lock(&pwq->pool->lock); aw\_spin\_lock(&pwq->pool->lock); f (wq->flags & WQ\_UNBOUND) { langle = langle ∃goto retry; lWARN\_ONCE(true, "workqueue: per-cpu pwq for %s on cpu%d has 0 wq->name, cpu); ce\_workqueue\_queue\_work(req\_cpu, pwq, work); owq->nr in flight[pwq->work color]++; work flags = work color to flags(pwg->work color); (likely(pwq->nr\_active < pwq->max\_active)) { cace\_workqueue\_activate\_work(work); vq->nr\_active++; orklist = &pwq->pool->worklist; f (list empty(worklist)) ]pwq->pool->watchdog\_ts = jiffies; rork\_flags |= WORK\_STRUCT\_DELAYED; orklist = &pwq->delayed\_works; ert\_work(pwq, work, worklist, work\_flags);

> ct pool\_workqueue { struct worker\_pool = \*pool; = /\* 1. the associated pool \*/ struct workqueue\_struct \*wq;□ int□work\_color; □/\* L: current color \*/ int□flush\_color;□/\* L: flushing color \*/ int□refcnt;□□/\* L: reference count \*/ int□nr\_in\_flight[WORK\_NR\_COLORS];/\* L: nr of in\_flight works \*/ int□nr\_active;□ /\* L: nr of active works \*/ int□max\_active; □ /\* L: max active works \*/ struct list\_head□delayed\_works; /\* L: delayed works \*/ struct list\_head□pwqs\_node;□ /\* WR: node on wq->pwqs \*/ struct list\_head□mayday\_node; /\* MD: node on wq->maydays \*/ struct work\_struct□unbound\_release\_work; struct rcu\_head□rcu;

\_queue\_work(int cpu, struct workqueue\_struct \*wq, struct work\_struct \*work)

aw spin unlock(&pwq->pool->lock);

u read unlock();

ruct worker\_pool { raw\_spinlock\_t□□lock;□/\* the pool lock \*/  $\operatorname{int} \square \square \square \operatorname{cpu}; \square / * I$ : the associated cpu  $* / \square$  $int \square \square \square node; \square/* I: the associated node ID */$  $\operatorname{int} \square \square \operatorname{id}; \square/* I: \operatorname{pool} \square */$ unsigned int□□flags;□/\* X: flags \*/ unsigned long□□watchdog\_ts;/\* L: watchdog timestamp \*/ struct list\_head□worklist;□/\* L: list of pending works \*/内核线程一直执行这里的 int□□□nr\_workers;□/\* L: total number of workers \*/ int□□□nr\_idle;□/\* L: currently idle workers \*/ struct list\_head□idle\_list;□/\* X: list of idle workers \*/ struct timer\_list□idle\_timer; □/\* L: worker idle timeout \*/ struct timer\_list□mayday\_timer;□/\* L: SOS timer for workers \*/ /\* a workers is either on busy\_hash or idle\_list, or the manager \*/ DECLARE\_HASHTABLE(busy\_hash, BUSY\_WORKER\_HASH\_ORDER); □□/\* L: hash of busy workers \*/ struct worker□□\*manager;□/\* L: purely informational \*/ struct list\_head□workers;□/\* A: attached workers \*/ struct completion□\*detach\_completion; /\* all workers detached \*/ struct ida□□worker\_ida;□/\* worker IDs for task name \*/ struct workqueue\_attrs□\*attrs;□/\* I: worker attributes \*/ struct hlist\_node□hash\_node;□/\* PL: unbound\_pool\_hash node \*/ int□□□refcnt;□/\* PL: refcnt for unbound pools \*/ atomic\_t□□nr\_running \_\_\_\_cacheline\_aligned\_in\_smp; struct rcu\_head□□rcu;

本文档是Linux内核编程04期:中断,的配套文档 结合视频教程、代码、PPT文档学习,效果更好。

视频地址: https://wanglitao.taobao.com

用户自创建workqueue defaule workqueue 前端接口: 创建工作队列 flag参数 后端实现: 创建线程池

workqueue workqueue