Go scheduler

本文基于Go 1.13。
Go提供了一套runtime机制,来提供其协程调度特性。实际上就是完成一种类似JavaScript的event-driven编程或者异步编程的范式。其带来的特性是,大大增加了Go app的并发特性(可达百万级,甚至更多的并发)。特别适用于所有并发都是无差别的场景下。例如网络连接。但因为没有优先级概念,实际上Go的调度框架对实时性是有损害的。
本文专注于分析调度框架,以及一些框架的实现细节。文末会列举我在阅读时参考的网页链接。

GPM

Go基于GPM这三个object来组织调度框架。大致结构如下:

  • G: Gourtines, 每个Goroutine对应一个G结构体,G保存Goroutine的运行堆栈,即并发任务状态。
  • P: Processors, 理想状态下一个P对应一个CPU核。对M来说,P提供了相关的执行环境(Context),如内存分配状态(mcache),任务队列(G)等
  • M: Machine, OS线程抽象,当和某个P绑定,从P的runq中不断取出G,切换堆栈并执行,M本身不具备执行状态,在需要任务切换时,M将堆栈状态写回G,任何其它M都能据此恢复执行。

schedule

调度始于Go bootstrap过程的mstart,每一轮调度,实际就是一次schedule函数的调用。
该函数定义在go/src/runtime/proc.go中。
本文针对调度器进行分析,仍然忽略以下特性,包括但不限于:

  • garbage collection
  • C Go
  • Linux signal处理
  • Trace操作
  • 栈操作
    一次调度的流程如下:
<svg id="djf6px01muo" width="100%" xmlns="http://www.w3.org/2000/svg" style="max-width: 300.3984375px;" viewBox="0 0 300.3984375 978.1984252929688"><style>


#djf6px01muo .label {
  font-family: 'trebuchet ms', verdana, arial;
  color: #333; }

#djf6px01muo .node rect,
#djf6px01muo .node circle,
#djf6px01muo .node ellipse,
#djf6px01muo .node polygon {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1px; }

#djf6px01muo .node.clickable {
  cursor: pointer; }

#djf6px01muo .arrowheadPath {
  fill: #333333; }

#djf6px01muo .edgePath .path {
  stroke: #333333;
  stroke-width: 1.5px; }

#djf6px01muo .edgeLabel {
  background-color: #e8e8e8; }

#djf6px01muo .cluster rect {
  fill: #ffffde !important;
  stroke: #aaaa33 !important;
  stroke-width: 1px !important; }

#djf6px01muo .cluster text {
  fill: #333; }

#djf6px01muo div.mermaidTooltip {
  position: absolute;
  text-align: center;
  max-width: 200px;
  padding: 2px;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 12px;
  background: #ffffde;
  border: 1px solid #aaaa33;
  border-radius: 2px;
  pointer-events: none;
  z-index: 100; }

#djf6px01muo .actor {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#djf6px01muo text.actor {
  fill: black;
  stroke: none; }

#djf6px01muo .actor-line {
  stroke: grey; }

#djf6px01muo .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#djf6px01muo .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#djf6px01muo #arrowhead {
  fill: #333; }

#djf6px01muo #crosshead path {
  fill: #333 !important;
  stroke: #333 !important; }

#djf6px01muo .messageText {
  fill: #333;
  stroke: none; }

#djf6px01muo .labelBox {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#djf6px01muo .labelText {
  fill: black;
  stroke: none; }

#djf6px01muo .loopText {
  fill: black;
  stroke: none; }

#djf6px01muo .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  stroke: #CCCCFF; }

#djf6px01muo .note {
  stroke: #aaaa33;
  fill: #fff5ad; }

#djf6px01muo .noteText {
  fill: black;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 14px; }

#djf6px01muo .activation0 {
  fill: #f4f4f4;
  stroke: #666; }

#djf6px01muo .activation1 {
  fill: #f4f4f4;
  stroke: #666; }

#djf6px01muo .activation2 {
  fill: #f4f4f4;
  stroke: #666; }


#djf6px01muo .section {
  stroke: none;
  opacity: 0.2; }

#djf6px01muo .section0 {
  fill: rgba(102, 102, 255, 0.49); }

#djf6px01muo .section2 {
  fill: #fff400; }

#djf6px01muo .section1,
#djf6px01muo .section3 {
  fill: white;
  opacity: 0.2; }

#djf6px01muo .sectionTitle0 {
  fill: #333; }

#djf6px01muo .sectionTitle1 {
  fill: #333; }

#djf6px01muo .sectionTitle2 {
  fill: #333; }

#djf6px01muo .sectionTitle3 {
  fill: #333; }

#djf6px01muo .sectionTitle {
  text-anchor: start;
  font-size: 11px;
  text-height: 14px; }


#djf6px01muo .grid .tick {
  stroke: lightgrey;
  opacity: 0.3;
  shape-rendering: crispEdges; }

#djf6px01muo .grid path {
  stroke-width: 0; }


#djf6px01muo .today {
  fill: none;
  stroke: red;
  stroke-width: 2px; }



#djf6px01muo .task {
  stroke-width: 2; }

#djf6px01muo .taskText {
  text-anchor: middle;
  font-size: 11px; }

#djf6px01muo .taskTextOutsideRight {
  fill: black;
  text-anchor: start;
  font-size: 11px; }

#djf6px01muo .taskTextOutsideLeft {
  fill: black;
  text-anchor: end;
  font-size: 11px; }


#djf6px01muo .taskText0,
#djf6px01muo .taskText1,
#djf6px01muo .taskText2,
#djf6px01muo .taskText3 {
  fill: white; }

#djf6px01muo .task0,
#djf6px01muo .task1,
#djf6px01muo .task2,
#djf6px01muo .task3 {
  fill: #8a90dd;
  stroke: #534fbc; }

#djf6px01muo .taskTextOutside0,
#djf6px01muo .taskTextOutside2 {
  fill: black; }

#djf6px01muo .taskTextOutside1,
#djf6px01muo .taskTextOutside3 {
  fill: black; }


#djf6px01muo .active0,
#djf6px01muo .active1,
#djf6px01muo .active2,
#djf6px01muo .active3 {
  fill: #bfc7ff;
  stroke: #534fbc; }

#djf6px01muo .activeText0,
#djf6px01muo .activeText1,
#djf6px01muo .activeText2,
#djf6px01muo .activeText3 {
  fill: black !important; }


#djf6px01muo .done0,
#djf6px01muo .done1,
#djf6px01muo .done2,
#djf6px01muo .done3 {
  stroke: grey;
  fill: lightgrey;
  stroke-width: 2; }

#djf6px01muo .doneText0,
#djf6px01muo .doneText1,
#djf6px01muo .doneText2,
#djf6px01muo .doneText3 {
  fill: black !important; }


#djf6px01muo .crit0,
#djf6px01muo .crit1,
#djf6px01muo .crit2,
#djf6px01muo .crit3 {
  stroke: #ff8888;
  fill: red;
  stroke-width: 2; }

#djf6px01muo .activeCrit0,
#djf6px01muo .activeCrit1,
#djf6px01muo .activeCrit2,
#djf6px01muo .activeCrit3 {
  stroke: #ff8888;
  fill: #bfc7ff;
  stroke-width: 2; }

#djf6px01muo .doneCrit0,
#djf6px01muo .doneCrit1,
#djf6px01muo .doneCrit2,
#djf6px01muo .doneCrit3 {
  stroke: #ff8888;
  fill: lightgrey;
  stroke-width: 2;
  cursor: pointer;
  shape-rendering: crispEdges; }

#djf6px01muo .doneCritText0,
#djf6px01muo .doneCritText1,
#djf6px01muo .doneCritText2,
#djf6px01muo .doneCritText3 {
  fill: black !important; }

#djf6px01muo .activeCritText0,
#djf6px01muo .activeCritText1,
#djf6px01muo .activeCritText2,
#djf6px01muo .activeCritText3 {
  fill: black !important; }

#djf6px01muo .titleText {
  text-anchor: middle;
  font-size: 18px;
  fill: black; }

#djf6px01muo g.classGroup text {
  fill: #9370DB;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 10px; }

#djf6px01muo g.classGroup rect {
  fill: #ECECFF;
  stroke: #9370DB; }

#djf6px01muo g.classGroup line {
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo .classLabel .box {
  stroke: none;
  stroke-width: 0;
  fill: #ECECFF;
  opacity: 0.5; }

#djf6px01muo .classLabel .label {
  fill: #9370DB;
  font-size: 10px; }

#djf6px01muo .relation {
  stroke: #9370DB;
  stroke-width: 1;
  fill: none; }

#djf6px01muo #compositionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo #compositionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo #aggregationStart {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo #aggregationEnd {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo #dependencyStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo #dependencyEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo #extensionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo #extensionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#djf6px01muo .commit-id,
#djf6px01muo .commit-msg,
#djf6px01muo .branch-label {
  fill: lightgrey;
  color: lightgrey; }



#djf6px01muo .label{
  color:#18B14E;
}
#djf6px01muo .te-md-container--dark .node rect {
  fill: red;
}

#djf6px01muo .node rect,
#djf6px01muo .node circle,
#djf6px01muo .node ellipse,
#djf6px01muo .node polygon {
  fill: #F9FFFB;;
  stroke: #2DBD60;
  stroke-width: 1.5px;
}
#djf6px01muo .arrowheadPath{
  fill: #2DBD60;
}
#djf6px01muo .edgePath .path {
  stroke: #2DBD60;
  stroke-width: 1px;
}
#djf6px01muo .edgeLabel {
  background-color: #fff;
}
#djf6px01muo .cluster rect {
  fill: #F9FFFB !important;
  stroke: #2DBD60 !important;
  stroke-width: 1px !important;
}

#djf6px01muo .cluster text {
  fill: #F9FFFB;
}

#djf6px01muo div.mermaidTooltip {
  background: #F9FFFB;
  border: 1px solid #2DBD60;
}


#djf6px01muo .actor {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#djf6px01muo text.actor {
  fill: #2DBD60;
  stroke: none;
}

#djf6px01muo .actor-line {
  stroke: #2DBD60;
}

#djf6px01muo .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#djf6px01muo .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #2DBD60;
}

#djf6px01muo #arrowhead {
  fill: #2DBD60;
}

#djf6px01muo #crosshead path {
  fill: #2DBD60 !important;
  stroke: #2DBD60 !important;
}

#djf6px01muo .messageText {
  fill: #2DBD60;
  stroke: none;
}

#djf6px01muo .labelBox {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#djf6px01muo .labelText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#djf6px01muo .loopText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#djf6px01muo .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#djf6px01muo .note {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#djf6px01muo .noteText {
  fill: #2DBD60;
  stroke: #2DBD60;
}


#djf6px01muo .section{
  opacity:1;
}
#djf6px01muo .section0,#djf6px01muo  .section2 {
  fill: #ECF7F0;
}

#djf6px01muo .section1,
#djf6px01muo .section3 {
  fill: #FFF;
}
#djf6px01muo .taskText0,
#djf6px01muo .taskText1,
#djf6px01muo .taskText2,
#djf6px01muo .taskText3 {
  fill: #fff;
}

#djf6px01muo .task0,
#djf6px01muo .task1,
#djf6px01muo .task2,
#djf6px01muo .task3 {
  fill: #2DBD60;
  stroke: #359F5A;
}
</style><style>#djf6px01muo {
    color: rgb(244, 244, 244);
    font: normal normal normal normal 14px/22.399999618530273px monospace;
  }</style><g transform="translate(-12, -12)"><g class="output"><g class="clusters"></g><g class="edgePaths"><g class="edgePath" style="opacity: 1;"><path class="path" d="M171.25,56.28125L171.25,81.28125L171.74999999999997,106.7812530517578" marker-end="url(#arrowhead9463)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9463" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M140.3842434285436,238.1748653767858L99.4296875,302.1812438964844L99.4296875,353.4624938964844L99.4296875,404.7437438964844L114.18493987088665,437.8843688964844" marker-end="url(#arrowhead9464)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9464" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M198.96450870977307,242.32611934198457L228.7890625,302.1812438964844L228.7890625,335.3218688964844" marker-end="url(#arrowhead9465)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9465" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M211.45956995543875,371.6031188964844L179.80078125,404.7437438964844L142.6160306215722,437.8843688964844" marker-end="url(#arrowhead9466)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9466" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M103.18006955553017,474.1656188964844L68.3203125,507.3062438964844L68.3203125,540.4468688964844" marker-end="url(#arrowhead9467)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9467" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M68.3203125,576.7281188964844L68.3203125,601.7281188964844L135.12942950735447,668.8018174408874" marker-end="url(#arrowhead9468)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9468" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M151.80742966420624,768.9589830089328L126.57421875,826.4953002929688L126.57421875,859.6359252929688" marker-end="url(#arrowhead9469)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9469" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M126.57421875,895.9171752929688L126.57421875,920.9171752929688L155.33417890936255,945.9171752929688" marker-end="url(#arrowhead9470)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9470" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M236.86584137911333,371.6031188964844L251.62109375,404.7437438964844L251.62109375,456.0249938964844L251.62109375,507.3062438964844L251.62109375,558.5874938964844L251.62109375,601.7281188964844L210.90154549807946,661.4265363428059" marker-end="url(#arrowhead9471)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9471" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M141.34336794446983,474.1656188964844L176.203125,507.3062438964844L176.203125,558.5874938964844L176.203125,601.7281188964844L176.70312499999997,627.2281219482421" marker-end="url(#arrowhead9472)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9472" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M197.893405323072,772.66440412517L215.92578125,826.4953002929688L215.92578125,877.7765502929688L215.92578125,920.9171752929688L192.90649475959796,945.9171752929688" marker-end="url(#arrowhead9473)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9473" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g></g><g class="edgeLabels"><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="translate(99.4296875,353.4624938964844)" style="opacity: 1;"><g transform="translate(-4.203125,-8.0078125)" class="label"><rect rx="0" ry="0" width="8.9375" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">N</tspan></text></g></g><g class="edgeLabel" transform="translate(228.7890625,302.1812438964844)" style="opacity: 1;"><g transform="translate(-4.203125,-8.0078125)" class="label"><rect rx="0" ry="0" width="8" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">Y</tspan></text></g></g><g class="edgeLabel" transform="translate(179.80078125,404.7437438964844)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="51.328125" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">gp == nil</tspan></text></g></g><g class="edgeLabel" transform="translate(68.3203125,507.3062438964844)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="51.328125" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">gp == nil</tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="translate(126.57421875,826.4953002929688)" style="opacity: 1;"><g transform="translate(-4.203125,-8.0078125)" class="label"><rect rx="0" ry="0" width="8" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">Y</tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="translate(251.62109375,507.3062438964844)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="49.140625" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">gp != nil</tspan></text></g></g><g class="edgeLabel" transform="translate(176.203125,558.5874938964844)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="49.140625" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">gp != nil</tspan></text></g></g><g class="edgeLabel" transform="translate(215.92578125,877.7765502929688)" style="opacity: 1;"><g transform="translate(-4.203125,-8.0078125)" class="label"><rect rx="0" ry="0" width="8.9375" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">N</tspan></text></g></g></g><g class="nodes"><g class="node" id="A" transform="translate(171.25,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-36.5" y="-18.140625" width="73" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-26.5,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">schedule</tspan></text></g></g></g><g class="node" id="B" transform="translate(171.25,187.6609344482422)" style="opacity: 1;"><polygon points="81.3796875,0 162.759375,-81.3796875 81.3796875,-162.759375 0,-81.3796875" rx="5" ry="5" transform="translate(-81.3796875,81.3796875)"></polygon><g class="label" transform="translate(0,0)"><g transform="translate(-62.28125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">schedtick % 61 == 0 ?</tspan></text></g></g></g><g class="node" id="D" transform="translate(122.26171875,456.0249938964844)" style="opacity: 1;"><rect rx="0" ry="0" x="-62.9765625" y="-18.140625" width="125.953125" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-52.9765625,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">runqget(_g_.m.p)</tspan></text></g></g></g><g class="node" id="C" transform="translate(228.7890625,353.4624938964844)" style="opacity: 1;"><rect rx="0" ry="0" x="-75.609375" y="-18.140625" width="151.21875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-65.609375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">globrunqget(_g_.m.p)</tspan></text></g></g></g><g class="node" id="E" transform="translate(68.3203125,558.5874938964844)" style="opacity: 1;"><rect rx="0" ry="0" x="-48.3203125" y="-18.140625" width="96.640625" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-38.3203125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">findrunnable</tspan></text></g></g></g><g class="node" id="G" transform="translate(176.203125,710.0413970947266)" style="opacity: 1;"><polygon points="83.31328125,0 166.6265625,-83.31328125 83.31328125,-166.6265625 0,-83.31328125" rx="5" ry="5" transform="translate(-83.31328125,83.31328125)"></polygon><g class="label" transform="translate(0,0)"><g transform="translate(-64.4296875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">_g_.m.spinning == 1 ?</tspan></text></g></g></g><g class="node" id="H" transform="translate(126.57421875,877.7765502929688)" style="opacity: 1;"><rect rx="0" ry="0" x="-49.8828125" y="-18.140625" width="99.765625" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-39.8828125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">resetspinning</tspan></text></g></g></g><g class="node" id="F" transform="translate(176.203125,964.0578002929688)" style="opacity: 1;"><rect rx="0" ry="0" x="-46.0859375" y="-18.140625" width="92.171875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-36.0859375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">execute(gp)</tspan></text></g></g></g></g></g></g></svg>

globrunqget & runqget

这两个函数非常类似,一个从全局队列中取,一个从本地队列中取。取的时机各有不同。

globrunqget

Go调度器为了全局队列中的G不被饿死,没61次调度都会检查一下全局队列中的G,并取出运行。

<svg id="d8yo634cgac" width="100%" xmlns="http://www.w3.org/2000/svg" style="max-width: 386.3125px;" viewBox="0 0 386.3125 586.25"><style>


#d8yo634cgac .label {
  font-family: 'trebuchet ms', verdana, arial;
  color: #333; }

#d8yo634cgac .node rect,
#d8yo634cgac .node circle,
#d8yo634cgac .node ellipse,
#d8yo634cgac .node polygon {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1px; }

#d8yo634cgac .node.clickable {
  cursor: pointer; }

#d8yo634cgac .arrowheadPath {
  fill: #333333; }

#d8yo634cgac .edgePath .path {
  stroke: #333333;
  stroke-width: 1.5px; }

#d8yo634cgac .edgeLabel {
  background-color: #e8e8e8; }

#d8yo634cgac .cluster rect {
  fill: #ffffde !important;
  stroke: #aaaa33 !important;
  stroke-width: 1px !important; }

#d8yo634cgac .cluster text {
  fill: #333; }

#d8yo634cgac div.mermaidTooltip {
  position: absolute;
  text-align: center;
  max-width: 200px;
  padding: 2px;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 12px;
  background: #ffffde;
  border: 1px solid #aaaa33;
  border-radius: 2px;
  pointer-events: none;
  z-index: 100; }

#d8yo634cgac .actor {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#d8yo634cgac text.actor {
  fill: black;
  stroke: none; }

#d8yo634cgac .actor-line {
  stroke: grey; }

#d8yo634cgac .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#d8yo634cgac .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#d8yo634cgac #arrowhead {
  fill: #333; }

#d8yo634cgac #crosshead path {
  fill: #333 !important;
  stroke: #333 !important; }

#d8yo634cgac .messageText {
  fill: #333;
  stroke: none; }

#d8yo634cgac .labelBox {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#d8yo634cgac .labelText {
  fill: black;
  stroke: none; }

#d8yo634cgac .loopText {
  fill: black;
  stroke: none; }

#d8yo634cgac .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  stroke: #CCCCFF; }

#d8yo634cgac .note {
  stroke: #aaaa33;
  fill: #fff5ad; }

#d8yo634cgac .noteText {
  fill: black;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 14px; }

#d8yo634cgac .activation0 {
  fill: #f4f4f4;
  stroke: #666; }

#d8yo634cgac .activation1 {
  fill: #f4f4f4;
  stroke: #666; }

#d8yo634cgac .activation2 {
  fill: #f4f4f4;
  stroke: #666; }


#d8yo634cgac .section {
  stroke: none;
  opacity: 0.2; }

#d8yo634cgac .section0 {
  fill: rgba(102, 102, 255, 0.49); }

#d8yo634cgac .section2 {
  fill: #fff400; }

#d8yo634cgac .section1,
#d8yo634cgac .section3 {
  fill: white;
  opacity: 0.2; }

#d8yo634cgac .sectionTitle0 {
  fill: #333; }

#d8yo634cgac .sectionTitle1 {
  fill: #333; }

#d8yo634cgac .sectionTitle2 {
  fill: #333; }

#d8yo634cgac .sectionTitle3 {
  fill: #333; }

#d8yo634cgac .sectionTitle {
  text-anchor: start;
  font-size: 11px;
  text-height: 14px; }


#d8yo634cgac .grid .tick {
  stroke: lightgrey;
  opacity: 0.3;
  shape-rendering: crispEdges; }

#d8yo634cgac .grid path {
  stroke-width: 0; }


#d8yo634cgac .today {
  fill: none;
  stroke: red;
  stroke-width: 2px; }



#d8yo634cgac .task {
  stroke-width: 2; }

#d8yo634cgac .taskText {
  text-anchor: middle;
  font-size: 11px; }

#d8yo634cgac .taskTextOutsideRight {
  fill: black;
  text-anchor: start;
  font-size: 11px; }

#d8yo634cgac .taskTextOutsideLeft {
  fill: black;
  text-anchor: end;
  font-size: 11px; }


#d8yo634cgac .taskText0,
#d8yo634cgac .taskText1,
#d8yo634cgac .taskText2,
#d8yo634cgac .taskText3 {
  fill: white; }

#d8yo634cgac .task0,
#d8yo634cgac .task1,
#d8yo634cgac .task2,
#d8yo634cgac .task3 {
  fill: #8a90dd;
  stroke: #534fbc; }

#d8yo634cgac .taskTextOutside0,
#d8yo634cgac .taskTextOutside2 {
  fill: black; }

#d8yo634cgac .taskTextOutside1,
#d8yo634cgac .taskTextOutside3 {
  fill: black; }


#d8yo634cgac .active0,
#d8yo634cgac .active1,
#d8yo634cgac .active2,
#d8yo634cgac .active3 {
  fill: #bfc7ff;
  stroke: #534fbc; }

#d8yo634cgac .activeText0,
#d8yo634cgac .activeText1,
#d8yo634cgac .activeText2,
#d8yo634cgac .activeText3 {
  fill: black !important; }


#d8yo634cgac .done0,
#d8yo634cgac .done1,
#d8yo634cgac .done2,
#d8yo634cgac .done3 {
  stroke: grey;
  fill: lightgrey;
  stroke-width: 2; }

#d8yo634cgac .doneText0,
#d8yo634cgac .doneText1,
#d8yo634cgac .doneText2,
#d8yo634cgac .doneText3 {
  fill: black !important; }


#d8yo634cgac .crit0,
#d8yo634cgac .crit1,
#d8yo634cgac .crit2,
#d8yo634cgac .crit3 {
  stroke: #ff8888;
  fill: red;
  stroke-width: 2; }

#d8yo634cgac .activeCrit0,
#d8yo634cgac .activeCrit1,
#d8yo634cgac .activeCrit2,
#d8yo634cgac .activeCrit3 {
  stroke: #ff8888;
  fill: #bfc7ff;
  stroke-width: 2; }

#d8yo634cgac .doneCrit0,
#d8yo634cgac .doneCrit1,
#d8yo634cgac .doneCrit2,
#d8yo634cgac .doneCrit3 {
  stroke: #ff8888;
  fill: lightgrey;
  stroke-width: 2;
  cursor: pointer;
  shape-rendering: crispEdges; }

#d8yo634cgac .doneCritText0,
#d8yo634cgac .doneCritText1,
#d8yo634cgac .doneCritText2,
#d8yo634cgac .doneCritText3 {
  fill: black !important; }

#d8yo634cgac .activeCritText0,
#d8yo634cgac .activeCritText1,
#d8yo634cgac .activeCritText2,
#d8yo634cgac .activeCritText3 {
  fill: black !important; }

#d8yo634cgac .titleText {
  text-anchor: middle;
  font-size: 18px;
  fill: black; }

#d8yo634cgac g.classGroup text {
  fill: #9370DB;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 10px; }

#d8yo634cgac g.classGroup rect {
  fill: #ECECFF;
  stroke: #9370DB; }

#d8yo634cgac g.classGroup line {
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac .classLabel .box {
  stroke: none;
  stroke-width: 0;
  fill: #ECECFF;
  opacity: 0.5; }

#d8yo634cgac .classLabel .label {
  fill: #9370DB;
  font-size: 10px; }

#d8yo634cgac .relation {
  stroke: #9370DB;
  stroke-width: 1;
  fill: none; }

#d8yo634cgac #compositionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac #compositionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac #aggregationStart {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac #aggregationEnd {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac #dependencyStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac #dependencyEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac #extensionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac #extensionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d8yo634cgac .commit-id,
#d8yo634cgac .commit-msg,
#d8yo634cgac .branch-label {
  fill: lightgrey;
  color: lightgrey; }



#d8yo634cgac .label{
  color:#18B14E;
}
#d8yo634cgac .te-md-container--dark .node rect {
  fill: red;
}

#d8yo634cgac .node rect,
#d8yo634cgac .node circle,
#d8yo634cgac .node ellipse,
#d8yo634cgac .node polygon {
  fill: #F9FFFB;;
  stroke: #2DBD60;
  stroke-width: 1.5px;
}
#d8yo634cgac .arrowheadPath{
  fill: #2DBD60;
}
#d8yo634cgac .edgePath .path {
  stroke: #2DBD60;
  stroke-width: 1px;
}
#d8yo634cgac .edgeLabel {
  background-color: #fff;
}
#d8yo634cgac .cluster rect {
  fill: #F9FFFB !important;
  stroke: #2DBD60 !important;
  stroke-width: 1px !important;
}

#d8yo634cgac .cluster text {
  fill: #F9FFFB;
}

#d8yo634cgac div.mermaidTooltip {
  background: #F9FFFB;
  border: 1px solid #2DBD60;
}


#d8yo634cgac .actor {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#d8yo634cgac text.actor {
  fill: #2DBD60;
  stroke: none;
}

#d8yo634cgac .actor-line {
  stroke: #2DBD60;
}

#d8yo634cgac .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#d8yo634cgac .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #2DBD60;
}

#d8yo634cgac #arrowhead {
  fill: #2DBD60;
}

#d8yo634cgac #crosshead path {
  fill: #2DBD60 !important;
  stroke: #2DBD60 !important;
}

#d8yo634cgac .messageText {
  fill: #2DBD60;
  stroke: none;
}

#d8yo634cgac .labelBox {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#d8yo634cgac .labelText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#d8yo634cgac .loopText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#d8yo634cgac .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#d8yo634cgac .note {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#d8yo634cgac .noteText {
  fill: #2DBD60;
  stroke: #2DBD60;
}


#d8yo634cgac .section{
  opacity:1;
}
#d8yo634cgac .section0,#d8yo634cgac  .section2 {
  fill: #ECF7F0;
}

#d8yo634cgac .section1,
#d8yo634cgac .section3 {
  fill: #FFF;
}
#d8yo634cgac .taskText0,
#d8yo634cgac .taskText1,
#d8yo634cgac .taskText2,
#d8yo634cgac .taskText3 {
  fill: #fff;
}

#d8yo634cgac .task0,
#d8yo634cgac .task1,
#d8yo634cgac .task2,
#d8yo634cgac .task3 {
  fill: #2DBD60;
  stroke: #359F5A;
}
</style><style>#d8yo634cgac {
    color: rgb(244, 244, 244);
    font: normal normal normal normal 14px/22.399999618530273px monospace;
  }</style><g transform="translate(-12, -12)"><g class="output"><g class="clusters"></g><g class="edgePaths"><g class="edgePath" style="opacity: 1;"><path class="path" d="M124.12039390996344,56.28125L59.671875,89.421875L59.671875,122.5625" marker-end="url(#arrowhead9508)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9508" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M194.67648109003656,56.28125L259.125,89.421875L259.125,122.5625" marker-end="url(#arrowhead9509)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9509" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M259.125,158.84375L259.125,183.84375L259.125,208.84375" marker-end="url(#arrowhead9510)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9510" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M259.125,245.125L259.125,270.125L259.125,295.125" marker-end="url(#arrowhead9511)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9511" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M259.125,331.40625L259.125,356.40625L259.125,381.40625" marker-end="url(#arrowhead9512)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9512" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M259.125,417.6875L259.125,442.6875L259.125,467.6875" marker-end="url(#arrowhead9513)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9513" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M259.125,503.96875L259.125,528.96875L259.125,553.96875" marker-end="url(#arrowhead9514)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9514" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g></g><g class="edgeLabels"><g class="edgeLabel" transform="translate(59.671875,89.421875)" style="opacity: 1;"><g transform="translate(-52.8125,-8.0078125)" class="label"><rect rx="0" ry="0" width="79.390625" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">runqsize == 0</tspan></text></g></g><g class="edgeLabel" transform="translate(259.125,89.421875)" style="opacity: 1;"><g transform="translate(-52.8125,-8.0078125)" class="label"><rect rx="0" ry="0" width="77.1875" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">runqsize != 0</tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g></g><g class="nodes"><g class="node" id="A" transform="translate(159.3984375,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-46.1171875" y="-18.140625" width="92.234375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-36.1171875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">globrunqget</tspan></text></g></g></g><g class="node" id="H" transform="translate(59.671875,140.703125)" style="opacity: 1;"><rect rx="0" ry="0" x="-31.2890625" y="-18.140625" width="62.578125" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-21.2890625,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">返回nil</tspan></text></g></g></g><g class="node" id="B" transform="translate(259.125,140.703125)" style="opacity: 1;"><rect rx="0" ry="0" x="-118.1640625" y="-18.140625" width="236.328125" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-108.1640625,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">n := sched.runqsize/gomaxprocs + 1</tspan></text></g></g></g><g class="node" id="C" transform="translate(259.125,226.984375)" style="opacity: 1;"><rect rx="0" ry="0" x="-54.9296875" y="-18.140625" width="109.859375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-44.9296875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">设置合适的n值</tspan></text></g></g></g><g class="node" id="D" transform="translate(259.125,313.265625)" style="opacity: 1;"><rect rx="0" ry="0" x="-48.7265625" y="-18.140625" width="97.453125" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-38.7265625,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">runqsize -= n</tspan></text></g></g></g><g class="node" id="E" transform="translate(259.125,399.546875)" style="opacity: 1;"><rect rx="0" ry="0" x="-99.4453125" y="-18.140625" width="198.890625" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-89.4453125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">从runq中pop G，z作为返回值</tspan></text></g></g></g><g class="node" id="F" transform="translate(259.125,485.828125)" style="opacity: 1;"><rect rx="0" ry="0" x="-131.1875" y="-18.140625" width="262.375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-121.1875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">从global runq中取出n个G，放入P的runq</tspan></text></g></g></g><g class="node" id="G" transform="translate(259.125,572.109375)" style="opacity: 1;"><rect rx="0" ry="0" x="-38.0078125" y="-18.140625" width="76.015625" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-28.0078125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">return gp</tspan></text></g></g></g></g></g></g></svg>

globrunqget的函数原型是:

func globrunqget(_p_ *p, max int32) *g

上图中n是要取出的G的个数。这里如果考虑max传入大于0的数,则n取值需要考虑:

  1. 不能超过runqsize,runqsize是此时global queue中G的个数
  2. 不能超过传入的max值
  3. 不能超过一个P的runqsize,该size定义在runtime2.go的type p声明中,当前值为256。
type p struct {
    ...
    runq     [256]guintptr
    ...
}

在schedule函数中globrunqget函数的max参数传入的是0,所以上述3个条件中的条件2就不用考虑了。

runqget

runqget会先查看P的runnext成员,如果有,则返回runnext指向的G
否则会取当前runqhead指向的runq数组的成员G。
runqhead和runqtail是两个int索引值。如果两值相等表示队列是空的。

findrunnable

findrunnable是一个阻塞型调用,来寻找runnable G。如果找不到,就会调用stopm,来休眠M。其基本流程如下:

<svg id="d859uuhkd6w" width="100%" xmlns="http://www.w3.org/2000/svg" style="max-width: 618.3046875px;" viewBox="0 0 618.3046875 807.65625"><style>


#d859uuhkd6w .label {
  font-family: 'trebuchet ms', verdana, arial;
  color: #333; }

#d859uuhkd6w .node rect,
#d859uuhkd6w .node circle,
#d859uuhkd6w .node ellipse,
#d859uuhkd6w .node polygon {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1px; }

#d859uuhkd6w .node.clickable {
  cursor: pointer; }

#d859uuhkd6w .arrowheadPath {
  fill: #333333; }

#d859uuhkd6w .edgePath .path {
  stroke: #333333;
  stroke-width: 1.5px; }

#d859uuhkd6w .edgeLabel {
  background-color: #e8e8e8; }

#d859uuhkd6w .cluster rect {
  fill: #ffffde !important;
  stroke: #aaaa33 !important;
  stroke-width: 1px !important; }

#d859uuhkd6w .cluster text {
  fill: #333; }

#d859uuhkd6w div.mermaidTooltip {
  position: absolute;
  text-align: center;
  max-width: 200px;
  padding: 2px;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 12px;
  background: #ffffde;
  border: 1px solid #aaaa33;
  border-radius: 2px;
  pointer-events: none;
  z-index: 100; }

#d859uuhkd6w .actor {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#d859uuhkd6w text.actor {
  fill: black;
  stroke: none; }

#d859uuhkd6w .actor-line {
  stroke: grey; }

#d859uuhkd6w .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#d859uuhkd6w .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#d859uuhkd6w #arrowhead {
  fill: #333; }

#d859uuhkd6w #crosshead path {
  fill: #333 !important;
  stroke: #333 !important; }

#d859uuhkd6w .messageText {
  fill: #333;
  stroke: none; }

#d859uuhkd6w .labelBox {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#d859uuhkd6w .labelText {
  fill: black;
  stroke: none; }

#d859uuhkd6w .loopText {
  fill: black;
  stroke: none; }

#d859uuhkd6w .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  stroke: #CCCCFF; }

#d859uuhkd6w .note {
  stroke: #aaaa33;
  fill: #fff5ad; }

#d859uuhkd6w .noteText {
  fill: black;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 14px; }

#d859uuhkd6w .activation0 {
  fill: #f4f4f4;
  stroke: #666; }

#d859uuhkd6w .activation1 {
  fill: #f4f4f4;
  stroke: #666; }

#d859uuhkd6w .activation2 {
  fill: #f4f4f4;
  stroke: #666; }


#d859uuhkd6w .section {
  stroke: none;
  opacity: 0.2; }

#d859uuhkd6w .section0 {
  fill: rgba(102, 102, 255, 0.49); }

#d859uuhkd6w .section2 {
  fill: #fff400; }

#d859uuhkd6w .section1,
#d859uuhkd6w .section3 {
  fill: white;
  opacity: 0.2; }

#d859uuhkd6w .sectionTitle0 {
  fill: #333; }

#d859uuhkd6w .sectionTitle1 {
  fill: #333; }

#d859uuhkd6w .sectionTitle2 {
  fill: #333; }

#d859uuhkd6w .sectionTitle3 {
  fill: #333; }

#d859uuhkd6w .sectionTitle {
  text-anchor: start;
  font-size: 11px;
  text-height: 14px; }


#d859uuhkd6w .grid .tick {
  stroke: lightgrey;
  opacity: 0.3;
  shape-rendering: crispEdges; }

#d859uuhkd6w .grid path {
  stroke-width: 0; }


#d859uuhkd6w .today {
  fill: none;
  stroke: red;
  stroke-width: 2px; }



#d859uuhkd6w .task {
  stroke-width: 2; }

#d859uuhkd6w .taskText {
  text-anchor: middle;
  font-size: 11px; }

#d859uuhkd6w .taskTextOutsideRight {
  fill: black;
  text-anchor: start;
  font-size: 11px; }

#d859uuhkd6w .taskTextOutsideLeft {
  fill: black;
  text-anchor: end;
  font-size: 11px; }


#d859uuhkd6w .taskText0,
#d859uuhkd6w .taskText1,
#d859uuhkd6w .taskText2,
#d859uuhkd6w .taskText3 {
  fill: white; }

#d859uuhkd6w .task0,
#d859uuhkd6w .task1,
#d859uuhkd6w .task2,
#d859uuhkd6w .task3 {
  fill: #8a90dd;
  stroke: #534fbc; }

#d859uuhkd6w .taskTextOutside0,
#d859uuhkd6w .taskTextOutside2 {
  fill: black; }

#d859uuhkd6w .taskTextOutside1,
#d859uuhkd6w .taskTextOutside3 {
  fill: black; }


#d859uuhkd6w .active0,
#d859uuhkd6w .active1,
#d859uuhkd6w .active2,
#d859uuhkd6w .active3 {
  fill: #bfc7ff;
  stroke: #534fbc; }

#d859uuhkd6w .activeText0,
#d859uuhkd6w .activeText1,
#d859uuhkd6w .activeText2,
#d859uuhkd6w .activeText3 {
  fill: black !important; }


#d859uuhkd6w .done0,
#d859uuhkd6w .done1,
#d859uuhkd6w .done2,
#d859uuhkd6w .done3 {
  stroke: grey;
  fill: lightgrey;
  stroke-width: 2; }

#d859uuhkd6w .doneText0,
#d859uuhkd6w .doneText1,
#d859uuhkd6w .doneText2,
#d859uuhkd6w .doneText3 {
  fill: black !important; }


#d859uuhkd6w .crit0,
#d859uuhkd6w .crit1,
#d859uuhkd6w .crit2,
#d859uuhkd6w .crit3 {
  stroke: #ff8888;
  fill: red;
  stroke-width: 2; }

#d859uuhkd6w .activeCrit0,
#d859uuhkd6w .activeCrit1,
#d859uuhkd6w .activeCrit2,
#d859uuhkd6w .activeCrit3 {
  stroke: #ff8888;
  fill: #bfc7ff;
  stroke-width: 2; }

#d859uuhkd6w .doneCrit0,
#d859uuhkd6w .doneCrit1,
#d859uuhkd6w .doneCrit2,
#d859uuhkd6w .doneCrit3 {
  stroke: #ff8888;
  fill: lightgrey;
  stroke-width: 2;
  cursor: pointer;
  shape-rendering: crispEdges; }

#d859uuhkd6w .doneCritText0,
#d859uuhkd6w .doneCritText1,
#d859uuhkd6w .doneCritText2,
#d859uuhkd6w .doneCritText3 {
  fill: black !important; }

#d859uuhkd6w .activeCritText0,
#d859uuhkd6w .activeCritText1,
#d859uuhkd6w .activeCritText2,
#d859uuhkd6w .activeCritText3 {
  fill: black !important; }

#d859uuhkd6w .titleText {
  text-anchor: middle;
  font-size: 18px;
  fill: black; }

#d859uuhkd6w g.classGroup text {
  fill: #9370DB;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 10px; }

#d859uuhkd6w g.classGroup rect {
  fill: #ECECFF;
  stroke: #9370DB; }

#d859uuhkd6w g.classGroup line {
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w .classLabel .box {
  stroke: none;
  stroke-width: 0;
  fill: #ECECFF;
  opacity: 0.5; }

#d859uuhkd6w .classLabel .label {
  fill: #9370DB;
  font-size: 10px; }

#d859uuhkd6w .relation {
  stroke: #9370DB;
  stroke-width: 1;
  fill: none; }

#d859uuhkd6w #compositionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w #compositionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w #aggregationStart {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w #aggregationEnd {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w #dependencyStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w #dependencyEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w #extensionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w #extensionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#d859uuhkd6w .commit-id,
#d859uuhkd6w .commit-msg,
#d859uuhkd6w .branch-label {
  fill: lightgrey;
  color: lightgrey; }



#d859uuhkd6w .label{
  color:#18B14E;
}
#d859uuhkd6w .te-md-container--dark .node rect {
  fill: red;
}

#d859uuhkd6w .node rect,
#d859uuhkd6w .node circle,
#d859uuhkd6w .node ellipse,
#d859uuhkd6w .node polygon {
  fill: #F9FFFB;;
  stroke: #2DBD60;
  stroke-width: 1.5px;
}
#d859uuhkd6w .arrowheadPath{
  fill: #2DBD60;
}
#d859uuhkd6w .edgePath .path {
  stroke: #2DBD60;
  stroke-width: 1px;
}
#d859uuhkd6w .edgeLabel {
  background-color: #fff;
}
#d859uuhkd6w .cluster rect {
  fill: #F9FFFB !important;
  stroke: #2DBD60 !important;
  stroke-width: 1px !important;
}

#d859uuhkd6w .cluster text {
  fill: #F9FFFB;
}

#d859uuhkd6w div.mermaidTooltip {
  background: #F9FFFB;
  border: 1px solid #2DBD60;
}


#d859uuhkd6w .actor {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#d859uuhkd6w text.actor {
  fill: #2DBD60;
  stroke: none;
}

#d859uuhkd6w .actor-line {
  stroke: #2DBD60;
}

#d859uuhkd6w .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#d859uuhkd6w .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #2DBD60;
}

#d859uuhkd6w #arrowhead {
  fill: #2DBD60;
}

#d859uuhkd6w #crosshead path {
  fill: #2DBD60 !important;
  stroke: #2DBD60 !important;
}

#d859uuhkd6w .messageText {
  fill: #2DBD60;
  stroke: none;
}

#d859uuhkd6w .labelBox {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#d859uuhkd6w .labelText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#d859uuhkd6w .loopText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#d859uuhkd6w .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#d859uuhkd6w .note {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#d859uuhkd6w .noteText {
  fill: #2DBD60;
  stroke: #2DBD60;
}


#d859uuhkd6w .section{
  opacity:1;
}
#d859uuhkd6w .section0,#d859uuhkd6w  .section2 {
  fill: #ECF7F0;
}

#d859uuhkd6w .section1,
#d859uuhkd6w .section3 {
  fill: #FFF;
}
#d859uuhkd6w .taskText0,
#d859uuhkd6w .taskText1,
#d859uuhkd6w .taskText2,
#d859uuhkd6w .taskText3 {
  fill: #fff;
}

#d859uuhkd6w .task0,
#d859uuhkd6w .task1,
#d859uuhkd6w .task2,
#d859uuhkd6w .task3 {
  fill: #2DBD60;
  stroke: #359F5A;
}
</style><style>#d859uuhkd6w {
    color: rgb(244, 244, 244);
    font: normal normal normal normal 14px/22.399999618530273px monospace;
  }</style><g transform="translate(-12, -12)"><g class="output"><g class="clusters"></g><g class="edgePaths"><g class="edgePath" style="opacity: 1;"><path class="path" d="M404.73828125,56.28125L404.73828125,81.28125L404.73828125,106.28125" marker-end="url(#arrowhead9608)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9608" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M355.62890625,135.58371494113572L179.11328125,175.703125L179.11328125,208.84375" marker-end="url(#arrowhead9609)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9609" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M355.62890625,131.26249459531869L36.5859375,175.703125L36.5859375,226.984375L36.5859375,278.265625L36.5859375,329.546875L36.5859375,380.828125L48.40781035763254,413.96875" marker-end="url(#arrowhead9610)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9610" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M200.82454724634368,245.125L240.48828125,278.265625L240.48828125,311.40625" marker-end="url(#arrowhead9611)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9611" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M254.71419032792505,347.6875L280.703125,380.828125L280.703125,413.96875" marker-end="url(#arrowhead9612)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9612" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M280.703125,450.25L280.703125,483.390625L333.7765939213894,516.53125" marker-end="url(#arrowhead9613)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9613" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M362.828125,552.8125L362.828125,577.8125L362.828125,602.8125" marker-end="url(#arrowhead9614)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9614" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M362.828125,639.09375L362.828125,664.09375L403.1418700199203,689.09375" marker-end="url(#arrowhead9615)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9615" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M432.39453125,725.375L432.39453125,750.375L466.0757156034951,775.375" marker-end="url(#arrowhead9616)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9616" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M518.9921875,782.2012216236149L599.09375,750.375L599.09375,707.234375L599.09375,664.09375L599.09375,620.953125L599.09375,577.8125L599.09375,534.671875L599.09375,483.390625L599.09375,432.109375L599.09375,380.828125L599.09375,329.546875L599.09375,278.265625L599.09375,226.984375L599.09375,175.703125L453.84765625,137.37952498492615" marker-end="url(#arrowhead9617)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9617" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M377.65374700068554,516.53125L404.73828125,483.390625L404.73828125,432.109375L404.73828125,380.828125L404.73828125,329.546875L404.73828125,278.265625L404.73828125,226.984375L404.73828125,175.703125L404.73828125,142.5625" marker-end="url(#arrowhead9618)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9618" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M466.4353056523904,689.09375L513.34765625,664.09375L513.34765625,620.953125L513.34765625,577.8125L513.34765625,534.671875L513.34765625,483.390625L513.34765625,432.109375L513.34765625,380.828125L513.34765625,329.546875L513.34765625,278.265625L513.34765625,226.984375L513.34765625,175.703125L443.1585994629799,142.5625" marker-end="url(#arrowhead9619)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9619" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M156.51626413962524,245.125L115.234375,278.265625L115.234375,329.546875L115.234375,380.828125L76.2295153964808,413.96875" marker-end="url(#arrowhead9620)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9620" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M224.3623600319927,347.6875L194.90234375,380.828125L83.81640625,421.51149797048487" marker-end="url(#arrowhead9621)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9621" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g></g><g class="edgeLabels"><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="translate(179.11328125,175.703125)" style="opacity: 1;"><g transform="translate(-16.359375,-8.0078125)" class="label"><rect rx="0" ry="0" width="28.15625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">no G</tspan></text></g></g><g class="edgeLabel" transform="translate(36.5859375,278.265625)" style="opacity: 1;"><g transform="translate(-20.40625,-8.0078125)" class="label"><rect rx="0" ry="0" width="33.40625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">has G</tspan></text></g></g><g class="edgeLabel" transform="translate(240.48828125,278.265625)" style="opacity: 1;"><g transform="translate(-16.359375,-8.0078125)" class="label"><rect rx="0" ry="0" width="28.15625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">no G</tspan></text></g></g><g class="edgeLabel" transform="translate(280.703125,380.828125)" style="opacity: 1;"><g transform="translate(-16.359375,-8.0078125)" class="label"><rect rx="0" ry="0" width="28.15625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">no G</tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="translate(599.09375,483.390625)" style="opacity: 1;"><g transform="translate(-24.4609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="46.765625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">wakeup</tspan></text></g></g><g class="edgeLabel" transform="translate(404.73828125,329.546875)" style="opacity: 1;"><g transform="translate(-77.1171875,-8.0078125)" class="label"><rect rx="0" ry="0" width="117.03125" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">sched.runqsize != 0</tspan></text></g></g><g class="edgeLabel" transform="translate(513.34765625,432.109375)" style="opacity: 1;"><g transform="translate(-40.6640625,-8.0078125)" class="label"><rect rx="0" ry="0" width="71.015625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">!runqempty</tspan></text></g></g><g class="edgeLabel" transform="translate(115.234375,329.546875)" style="opacity: 1;"><g transform="translate(-20.40625,-8.0078125)" class="label"><rect rx="0" ry="0" width="33.40625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">has G</tspan></text></g></g><g class="edgeLabel" transform="translate(194.90234375,380.828125)" style="opacity: 1;"><g transform="translate(-20.40625,-8.0078125)" class="label"><rect rx="0" ry="0" width="33.40625" height="16.390625" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">has G</tspan></text></g></g></g><g class="nodes"><g class="node" id="A" transform="translate(404.73828125,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-48.3203125" y="-18.140625" width="96.640625" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-38.3203125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">findrunnable</tspan></text></g></g></g><g class="node" id="B" transform="translate(404.73828125,124.421875)" style="opacity: 1;"><rect rx="0" ry="0" x="-49.109375" y="-18.140625" width="98.21875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-39.109375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">runqget(_p_)</tspan></text></g></g></g><g class="node" id="C" transform="translate(179.11328125,226.984375)" style="opacity: 1;"><rect rx="0" ry="0" x="-85.7109375" y="-18.140625" width="171.421875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-75.7109375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">globrunqget(_p_, max=0)</tspan></text></g></g></g><g class="node" id="J" transform="translate(54.87890625,432.109375)" style="opacity: 1;"><rect rx="0" ry="0" x="-28.9375" y="-18.140625" width="57.875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-18.9375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">return</tspan></text></g></g></g><g class="node" id="D" transform="translate(240.48828125,329.546875)" style="opacity: 1;"><rect rx="0" ry="0" x="-71.1640625" y="-18.140625" width="142.328125" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-61.1640625,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">steal G from other P</tspan></text></g></g></g><g class="node" id="E" transform="translate(280.703125,432.109375)" style="opacity: 1;"><rect rx="0" ry="0" x="-48.8203125" y="-18.140625" width="97.640625" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-38.8203125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">going to stop</tspan></text></g></g></g><g class="node" id="F" transform="translate(362.828125,534.671875)" style="opacity: 1;"><rect rx="0" ry="0" x="-104.1328125" y="-18.140625" width="208.265625" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-94.1328125,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">before stop, check globrunqget</tspan></text></g></g></g><g class="node" id="G" transform="translate(362.828125,620.953125)" style="opacity: 1;"><rect rx="0" ry="0" x="-50.765625" y="-18.140625" width="101.53125" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-40.765625,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">pidleput(_p_)</tspan></text></g></g></g><g class="node" id="H" transform="translate(432.39453125,707.234375)" style="opacity: 1;"><rect rx="0" ry="0" x="-81.2421875" y="-18.140625" width="162.484375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-71.2421875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">before stop, check all P</tspan></text></g></g></g><g class="node" id="I" transform="translate(490.515625,793.515625)" style="opacity: 1;"><rect rx="0" ry="0" x="-28.4765625" y="-18.140625" width="56.953125" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-18.4765625,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">stopm</tspan></text></g></g></g></g></g></g></svg>

这里忽略了net poller,这是用来处理异步系统调用的一种方式。
以上流程中除了常规的global q和local q操作,还有一段关于stealing G的算法,称为Go Stealing Scheduler,可以参考Go's work-stealing scheduler。具体代码如下:

for i := 0; i < 4; i++ {
    for enum := stealOrder.start(fastrand()); !enum.done(); enum.next() {
        if sched.gcwaiting != 0 {
            goto top
        }
        stealRunNextG := i > 2 // first look for ready queues with more than 1 g
        if gp := runqsteal(_p_, allp[enum.position()], stealRunNextG); gp != nil {
            return gp, false
        }
    }
}

说实话代码没有很看懂:

  • 这个i = [0..4]是怎么回事儿?
  • 第二个for循环似乎采用随机数来随机获得相应的P,来steal G,但是为啥有个stealRunNextG := i > 2

stealRunNextG根据代码理解,应该是bool变量,表示是否从P的runnext成员获取G。
runqsteal会从runnext偷取G,或者会偷取P的runq中一半的G

resetspinning

这个函数将原先为true的_g_.m.spinning设为false。当全局都没有spinning的M了,则要唤醒idle P。没有spinning M表示所有的M要么在工作(处理G),要么在休眠。如果此时有idle P,表示有CPU空闲,则此时是一个好的时机来唤醒M处理更多的工作。
实际上GPM三者都有自己的状态机,G和P比较清晰,M的状态就比较模糊,到底M有几种状态,特别的是M会在何时spinning。后文会介绍。

execute

简单来看,execute只做了两件事:

  1. 切换G status:_Grunnalbe --> _Grunning
  2. 调用gogo(&gp.sched)

ARM平台的gogo函数实现在:go/src/runtime/asm_arm.s,是一个汇编函数。主要做了以下几件事情:

  • 设置TLS中的G
  • 做context switch
  • 跳转到对应的PC取运行

GPM的状态变迁

GPM的状态变迁可以参考Golang 中 goroutine 的调度一文中关于GPM的一生的章节。

G & P

对G和P不做过多的解释,状态机还是很清晰的。
G由newproc函数创建,遵循下面的状态变化

P由bootstrap过程的procresize创建,遵循下面的状态机变迁:

M的状态

这里不讨论特殊的M,例如:sysmon或者finalizer
普通的M由startm函数创建。其大致流程如下:

<svg id="dvbkimlurdk" width="100%" xmlns="http://www.w3.org/2000/svg" style="max-width: 357.7109375px;" viewBox="0 0 357.7109375 516.25"><style>


#dvbkimlurdk .label {
  font-family: 'trebuchet ms', verdana, arial;
  color: #333; }

#dvbkimlurdk .node rect,
#dvbkimlurdk .node circle,
#dvbkimlurdk .node ellipse,
#dvbkimlurdk .node polygon {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1px; }

#dvbkimlurdk .node.clickable {
  cursor: pointer; }

#dvbkimlurdk .arrowheadPath {
  fill: #333333; }

#dvbkimlurdk .edgePath .path {
  stroke: #333333;
  stroke-width: 1.5px; }

#dvbkimlurdk .edgeLabel {
  background-color: #e8e8e8; }

#dvbkimlurdk .cluster rect {
  fill: #ffffde !important;
  stroke: #aaaa33 !important;
  stroke-width: 1px !important; }

#dvbkimlurdk .cluster text {
  fill: #333; }

#dvbkimlurdk div.mermaidTooltip {
  position: absolute;
  text-align: center;
  max-width: 200px;
  padding: 2px;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 12px;
  background: #ffffde;
  border: 1px solid #aaaa33;
  border-radius: 2px;
  pointer-events: none;
  z-index: 100; }

#dvbkimlurdk .actor {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#dvbkimlurdk text.actor {
  fill: black;
  stroke: none; }

#dvbkimlurdk .actor-line {
  stroke: grey; }

#dvbkimlurdk .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#dvbkimlurdk .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#dvbkimlurdk #arrowhead {
  fill: #333; }

#dvbkimlurdk #crosshead path {
  fill: #333 !important;
  stroke: #333 !important; }

#dvbkimlurdk .messageText {
  fill: #333;
  stroke: none; }

#dvbkimlurdk .labelBox {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#dvbkimlurdk .labelText {
  fill: black;
  stroke: none; }

#dvbkimlurdk .loopText {
  fill: black;
  stroke: none; }

#dvbkimlurdk .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  stroke: #CCCCFF; }

#dvbkimlurdk .note {
  stroke: #aaaa33;
  fill: #fff5ad; }

#dvbkimlurdk .noteText {
  fill: black;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 14px; }

#dvbkimlurdk .activation0 {
  fill: #f4f4f4;
  stroke: #666; }

#dvbkimlurdk .activation1 {
  fill: #f4f4f4;
  stroke: #666; }

#dvbkimlurdk .activation2 {
  fill: #f4f4f4;
  stroke: #666; }


#dvbkimlurdk .section {
  stroke: none;
  opacity: 0.2; }

#dvbkimlurdk .section0 {
  fill: rgba(102, 102, 255, 0.49); }

#dvbkimlurdk .section2 {
  fill: #fff400; }

#dvbkimlurdk .section1,
#dvbkimlurdk .section3 {
  fill: white;
  opacity: 0.2; }

#dvbkimlurdk .sectionTitle0 {
  fill: #333; }

#dvbkimlurdk .sectionTitle1 {
  fill: #333; }

#dvbkimlurdk .sectionTitle2 {
  fill: #333; }

#dvbkimlurdk .sectionTitle3 {
  fill: #333; }

#dvbkimlurdk .sectionTitle {
  text-anchor: start;
  font-size: 11px;
  text-height: 14px; }


#dvbkimlurdk .grid .tick {
  stroke: lightgrey;
  opacity: 0.3;
  shape-rendering: crispEdges; }

#dvbkimlurdk .grid path {
  stroke-width: 0; }


#dvbkimlurdk .today {
  fill: none;
  stroke: red;
  stroke-width: 2px; }



#dvbkimlurdk .task {
  stroke-width: 2; }

#dvbkimlurdk .taskText {
  text-anchor: middle;
  font-size: 11px; }

#dvbkimlurdk .taskTextOutsideRight {
  fill: black;
  text-anchor: start;
  font-size: 11px; }

#dvbkimlurdk .taskTextOutsideLeft {
  fill: black;
  text-anchor: end;
  font-size: 11px; }


#dvbkimlurdk .taskText0,
#dvbkimlurdk .taskText1,
#dvbkimlurdk .taskText2,
#dvbkimlurdk .taskText3 {
  fill: white; }

#dvbkimlurdk .task0,
#dvbkimlurdk .task1,
#dvbkimlurdk .task2,
#dvbkimlurdk .task3 {
  fill: #8a90dd;
  stroke: #534fbc; }

#dvbkimlurdk .taskTextOutside0,
#dvbkimlurdk .taskTextOutside2 {
  fill: black; }

#dvbkimlurdk .taskTextOutside1,
#dvbkimlurdk .taskTextOutside3 {
  fill: black; }


#dvbkimlurdk .active0,
#dvbkimlurdk .active1,
#dvbkimlurdk .active2,
#dvbkimlurdk .active3 {
  fill: #bfc7ff;
  stroke: #534fbc; }

#dvbkimlurdk .activeText0,
#dvbkimlurdk .activeText1,
#dvbkimlurdk .activeText2,
#dvbkimlurdk .activeText3 {
  fill: black !important; }


#dvbkimlurdk .done0,
#dvbkimlurdk .done1,
#dvbkimlurdk .done2,
#dvbkimlurdk .done3 {
  stroke: grey;
  fill: lightgrey;
  stroke-width: 2; }

#dvbkimlurdk .doneText0,
#dvbkimlurdk .doneText1,
#dvbkimlurdk .doneText2,
#dvbkimlurdk .doneText3 {
  fill: black !important; }


#dvbkimlurdk .crit0,
#dvbkimlurdk .crit1,
#dvbkimlurdk .crit2,
#dvbkimlurdk .crit3 {
  stroke: #ff8888;
  fill: red;
  stroke-width: 2; }

#dvbkimlurdk .activeCrit0,
#dvbkimlurdk .activeCrit1,
#dvbkimlurdk .activeCrit2,
#dvbkimlurdk .activeCrit3 {
  stroke: #ff8888;
  fill: #bfc7ff;
  stroke-width: 2; }

#dvbkimlurdk .doneCrit0,
#dvbkimlurdk .doneCrit1,
#dvbkimlurdk .doneCrit2,
#dvbkimlurdk .doneCrit3 {
  stroke: #ff8888;
  fill: lightgrey;
  stroke-width: 2;
  cursor: pointer;
  shape-rendering: crispEdges; }

#dvbkimlurdk .doneCritText0,
#dvbkimlurdk .doneCritText1,
#dvbkimlurdk .doneCritText2,
#dvbkimlurdk .doneCritText3 {
  fill: black !important; }

#dvbkimlurdk .activeCritText0,
#dvbkimlurdk .activeCritText1,
#dvbkimlurdk .activeCritText2,
#dvbkimlurdk .activeCritText3 {
  fill: black !important; }

#dvbkimlurdk .titleText {
  text-anchor: middle;
  font-size: 18px;
  fill: black; }

#dvbkimlurdk g.classGroup text {
  fill: #9370DB;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 10px; }

#dvbkimlurdk g.classGroup rect {
  fill: #ECECFF;
  stroke: #9370DB; }

#dvbkimlurdk g.classGroup line {
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk .classLabel .box {
  stroke: none;
  stroke-width: 0;
  fill: #ECECFF;
  opacity: 0.5; }

#dvbkimlurdk .classLabel .label {
  fill: #9370DB;
  font-size: 10px; }

#dvbkimlurdk .relation {
  stroke: #9370DB;
  stroke-width: 1;
  fill: none; }

#dvbkimlurdk #compositionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk #compositionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk #aggregationStart {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk #aggregationEnd {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk #dependencyStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk #dependencyEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk #extensionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk #extensionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#dvbkimlurdk .commit-id,
#dvbkimlurdk .commit-msg,
#dvbkimlurdk .branch-label {
  fill: lightgrey;
  color: lightgrey; }



#dvbkimlurdk .label{
  color:#18B14E;
}
#dvbkimlurdk .te-md-container--dark .node rect {
  fill: red;
}

#dvbkimlurdk .node rect,
#dvbkimlurdk .node circle,
#dvbkimlurdk .node ellipse,
#dvbkimlurdk .node polygon {
  fill: #F9FFFB;;
  stroke: #2DBD60;
  stroke-width: 1.5px;
}
#dvbkimlurdk .arrowheadPath{
  fill: #2DBD60;
}
#dvbkimlurdk .edgePath .path {
  stroke: #2DBD60;
  stroke-width: 1px;
}
#dvbkimlurdk .edgeLabel {
  background-color: #fff;
}
#dvbkimlurdk .cluster rect {
  fill: #F9FFFB !important;
  stroke: #2DBD60 !important;
  stroke-width: 1px !important;
}

#dvbkimlurdk .cluster text {
  fill: #F9FFFB;
}

#dvbkimlurdk div.mermaidTooltip {
  background: #F9FFFB;
  border: 1px solid #2DBD60;
}


#dvbkimlurdk .actor {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#dvbkimlurdk text.actor {
  fill: #2DBD60;
  stroke: none;
}

#dvbkimlurdk .actor-line {
  stroke: #2DBD60;
}

#dvbkimlurdk .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#dvbkimlurdk .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #2DBD60;
}

#dvbkimlurdk #arrowhead {
  fill: #2DBD60;
}

#dvbkimlurdk #crosshead path {
  fill: #2DBD60 !important;
  stroke: #2DBD60 !important;
}

#dvbkimlurdk .messageText {
  fill: #2DBD60;
  stroke: none;
}

#dvbkimlurdk .labelBox {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#dvbkimlurdk .labelText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#dvbkimlurdk .loopText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#dvbkimlurdk .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#dvbkimlurdk .note {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#dvbkimlurdk .noteText {
  fill: #2DBD60;
  stroke: #2DBD60;
}


#dvbkimlurdk .section{
  opacity:1;
}
#dvbkimlurdk .section0,#dvbkimlurdk  .section2 {
  fill: #ECF7F0;
}

#dvbkimlurdk .section1,
#dvbkimlurdk .section3 {
  fill: #FFF;
}
#dvbkimlurdk .taskText0,
#dvbkimlurdk .taskText1,
#dvbkimlurdk .taskText2,
#dvbkimlurdk .taskText3 {
  fill: #fff;
}

#dvbkimlurdk .task0,
#dvbkimlurdk .task1,
#dvbkimlurdk .task2,
#dvbkimlurdk .task3 {
  fill: #2DBD60;
  stroke: #359F5A;
}
</style><style>#dvbkimlurdk {
    color: rgb(244, 244, 244);
    font: normal normal normal normal 14px/22.399999618530273px monospace;
  }</style><g transform="translate(-12, -12)"><g class="output"><g class="clusters"></g><g class="edgePaths"><g class="edgePath" style="opacity: 1;"><path class="path" d="M130.84453553473492,56.28125L71.63671875,89.421875L71.63671875,122.5625" marker-end="url(#arrowhead9666)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9666" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M84.76822083523766,158.84375L108.7578125,191.984375L174.90625,229.00990102967512" marker-end="url(#arrowhead9667)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9667" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M185.35484746343693,56.28125L225.73046875,89.421875L225.73046875,140.703125L225.73046875,191.984375L209.34443912819927,225.125" marker-end="url(#arrowhead9668)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9668" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M62.66727962180073,158.84375L46.28125,191.984375L46.28125,243.265625L46.28125,286.40625L46.28125,329.546875L46.28125,372.6875L46.28125,415.828125L46.28125,458.96875L91.5859375,485.2943823660949" marker-end="url(#arrowhead9669)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9669" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M174.90625,257.02537820418746L120.5234375,286.40625L120.5234375,329.546875L120.5234375,372.6875L120.5234375,397.6875" marker-end="url(#arrowhead9670)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9670" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M120.5234375,433.96875L120.5234375,458.96875L120.5234375,483.96875" marker-end="url(#arrowhead9671)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9671" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M225.84375,257.02537820418746L280.2265625,286.40625L280.2265625,311.40625" marker-end="url(#arrowhead9672)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9672" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M280.2265625,347.6875L280.2265625,372.6875L280.2265625,397.6875" marker-end="url(#arrowhead9673)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9673" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M280.2265625,433.96875L280.2265625,458.96875L149.4609375,494.29248452940027" marker-end="url(#arrowhead9674)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9674" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g></g><g class="edgeLabels"><g class="edgeLabel" transform="translate(71.63671875,89.421875)" style="opacity: 1;"><g transform="translate(-40.6640625,-8.0078125)" class="label"><rect rx="0" ry="0" width="58.703125" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">_p_ == nil</tspan></text></g></g><g class="edgeLabel" transform="translate(108.7578125,191.984375)" style="opacity: 1;"><g transform="translate(-44.7109375,-8.0078125)" class="label"><rect rx="0" ry="0" width="61.453125" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">find idle P</tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="translate(46.28125,329.546875)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="52.59375" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">no idle P</tspan></text></g></g><g class="edgeLabel" transform="translate(120.5234375,329.546875)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="55.953125" height="16.28125" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">mp == nil</tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g><g class="edgeLabel" transform="" style="opacity: 1;"><g transform="translate(0,0)" class="label"><rect rx="0" ry="0" width="0" height="0" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1"></tspan></text></g></g></g><g class="nodes"><g class="node" id="A" transform="translate(163.25390625,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-74.375" y="-18.140625" width="148.75" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-64.375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">startm(_p_, spinning)</tspan></text></g></g></g><g class="node" id="B" transform="translate(71.63671875,140.703125)" style="opacity: 1;"><rect rx="0" ry="0" x="-34.7421875" y="-18.140625" width="69.484375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-24.7421875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">pidleget</tspan></text></g></g></g><g class="node" id="C" transform="translate(200.375,243.265625)" style="opacity: 1;"><rect rx="0" ry="0" x="-25.46875" y="-18.140625" width="50.9375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-15.46875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">mget</tspan></text></g></g></g><g class="node" id="G" transform="translate(120.5234375,502.109375)" style="opacity: 1;"><rect rx="0" ry="0" x="-28.9375" y="-18.140625" width="57.875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-18.9375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">return</tspan></text></g></g></g><g class="node" id="D" transform="translate(120.5234375,415.828125)" style="opacity: 1;"><rect rx="0" ry="0" x="-28.21875" y="-18.140625" width="56.4375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-18.21875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">newm</tspan></text></g></g></g><g class="node" id="E" transform="translate(280.2265625,329.546875)" style="opacity: 1;"><rect rx="0" ry="0" x="-66.0859375" y="-18.140625" width="132.171875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-56.0859375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">mp.nextp.set(_p_)</tspan></text></g></g></g><g class="node" id="F" transform="translate(280.2265625,415.828125)" style="opacity: 1;"><rect rx="0" ry="0" x="-81.484375" y="-18.140625" width="162.96875" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-71.484375,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">notewakeup(&amp;mp.park)</tspan></text></g></g></g></g></g></g></svg>
M的创建

这里跟M状态有关的就是M的线程函数,但请注意下面这个fn并不是线程函数。

if spinning {
    // The caller incremented nmspinning, so set m.spinning in the new M.
    fn = mspinning
}

真正的线程函数在newosproc调用clone接口的时候传入。Linux平台的实现在os_linux.go中。如下:

func clone(flags int32, stk, mp, gp, fn unsafe.Pointer) int32

func newosproc(mp *m) {
    ...
	ret := clone(cloneFlags, stk, unsafe.Pointer(mp), unsafe.Pointer(mp.g0), unsafe.Pointer(funcPC(mstart)))
    ...
}

所以M的线程函数就是曾在bootstrap中出现过,运行M0的mstart函数。上面的mspinning函数在mstart1中被调用,它将线程设置为spinning状态。

idle M的管理

mget会去获取闲置的M。而所有的idle M并没有一个统一的保存位置,而是通过M.schedlink形成一个隐形的链表。看mput:

func mput(mp *m) {
	mp.schedlink = sched.midle
	sched.midle.set(mp)
	sched.nmidle++
    ...
}

假设调用了3次mput,那么此时的idle链表应该是这样的:

<svg id="du9tpl1wbp9" width="100%" xmlns="http://www.w3.org/2000/svg" style="max-width: 611.140625px;" viewBox="0 0 611.140625 52.28125"><style>


#du9tpl1wbp9 .label {
  font-family: 'trebuchet ms', verdana, arial;
  color: #333; }

#du9tpl1wbp9 .node rect,
#du9tpl1wbp9 .node circle,
#du9tpl1wbp9 .node ellipse,
#du9tpl1wbp9 .node polygon {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1px; }

#du9tpl1wbp9 .node.clickable {
  cursor: pointer; }

#du9tpl1wbp9 .arrowheadPath {
  fill: #333333; }

#du9tpl1wbp9 .edgePath .path {
  stroke: #333333;
  stroke-width: 1.5px; }

#du9tpl1wbp9 .edgeLabel {
  background-color: #e8e8e8; }

#du9tpl1wbp9 .cluster rect {
  fill: #ffffde !important;
  stroke: #aaaa33 !important;
  stroke-width: 1px !important; }

#du9tpl1wbp9 .cluster text {
  fill: #333; }

#du9tpl1wbp9 div.mermaidTooltip {
  position: absolute;
  text-align: center;
  max-width: 200px;
  padding: 2px;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 12px;
  background: #ffffde;
  border: 1px solid #aaaa33;
  border-radius: 2px;
  pointer-events: none;
  z-index: 100; }

#du9tpl1wbp9 .actor {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#du9tpl1wbp9 text.actor {
  fill: black;
  stroke: none; }

#du9tpl1wbp9 .actor-line {
  stroke: grey; }

#du9tpl1wbp9 .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#du9tpl1wbp9 .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #333; }

#du9tpl1wbp9 #arrowhead {
  fill: #333; }

#du9tpl1wbp9 #crosshead path {
  fill: #333 !important;
  stroke: #333 !important; }

#du9tpl1wbp9 .messageText {
  fill: #333;
  stroke: none; }

#du9tpl1wbp9 .labelBox {
  stroke: #CCCCFF;
  fill: #ECECFF; }

#du9tpl1wbp9 .labelText {
  fill: black;
  stroke: none; }

#du9tpl1wbp9 .loopText {
  fill: black;
  stroke: none; }

#du9tpl1wbp9 .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  stroke: #CCCCFF; }

#du9tpl1wbp9 .note {
  stroke: #aaaa33;
  fill: #fff5ad; }

#du9tpl1wbp9 .noteText {
  fill: black;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 14px; }

#du9tpl1wbp9 .activation0 {
  fill: #f4f4f4;
  stroke: #666; }

#du9tpl1wbp9 .activation1 {
  fill: #f4f4f4;
  stroke: #666; }

#du9tpl1wbp9 .activation2 {
  fill: #f4f4f4;
  stroke: #666; }


#du9tpl1wbp9 .section {
  stroke: none;
  opacity: 0.2; }

#du9tpl1wbp9 .section0 {
  fill: rgba(102, 102, 255, 0.49); }

#du9tpl1wbp9 .section2 {
  fill: #fff400; }

#du9tpl1wbp9 .section1,
#du9tpl1wbp9 .section3 {
  fill: white;
  opacity: 0.2; }

#du9tpl1wbp9 .sectionTitle0 {
  fill: #333; }

#du9tpl1wbp9 .sectionTitle1 {
  fill: #333; }

#du9tpl1wbp9 .sectionTitle2 {
  fill: #333; }

#du9tpl1wbp9 .sectionTitle3 {
  fill: #333; }

#du9tpl1wbp9 .sectionTitle {
  text-anchor: start;
  font-size: 11px;
  text-height: 14px; }


#du9tpl1wbp9 .grid .tick {
  stroke: lightgrey;
  opacity: 0.3;
  shape-rendering: crispEdges; }

#du9tpl1wbp9 .grid path {
  stroke-width: 0; }


#du9tpl1wbp9 .today {
  fill: none;
  stroke: red;
  stroke-width: 2px; }



#du9tpl1wbp9 .task {
  stroke-width: 2; }

#du9tpl1wbp9 .taskText {
  text-anchor: middle;
  font-size: 11px; }

#du9tpl1wbp9 .taskTextOutsideRight {
  fill: black;
  text-anchor: start;
  font-size: 11px; }

#du9tpl1wbp9 .taskTextOutsideLeft {
  fill: black;
  text-anchor: end;
  font-size: 11px; }


#du9tpl1wbp9 .taskText0,
#du9tpl1wbp9 .taskText1,
#du9tpl1wbp9 .taskText2,
#du9tpl1wbp9 .taskText3 {
  fill: white; }

#du9tpl1wbp9 .task0,
#du9tpl1wbp9 .task1,
#du9tpl1wbp9 .task2,
#du9tpl1wbp9 .task3 {
  fill: #8a90dd;
  stroke: #534fbc; }

#du9tpl1wbp9 .taskTextOutside0,
#du9tpl1wbp9 .taskTextOutside2 {
  fill: black; }

#du9tpl1wbp9 .taskTextOutside1,
#du9tpl1wbp9 .taskTextOutside3 {
  fill: black; }


#du9tpl1wbp9 .active0,
#du9tpl1wbp9 .active1,
#du9tpl1wbp9 .active2,
#du9tpl1wbp9 .active3 {
  fill: #bfc7ff;
  stroke: #534fbc; }

#du9tpl1wbp9 .activeText0,
#du9tpl1wbp9 .activeText1,
#du9tpl1wbp9 .activeText2,
#du9tpl1wbp9 .activeText3 {
  fill: black !important; }


#du9tpl1wbp9 .done0,
#du9tpl1wbp9 .done1,
#du9tpl1wbp9 .done2,
#du9tpl1wbp9 .done3 {
  stroke: grey;
  fill: lightgrey;
  stroke-width: 2; }

#du9tpl1wbp9 .doneText0,
#du9tpl1wbp9 .doneText1,
#du9tpl1wbp9 .doneText2,
#du9tpl1wbp9 .doneText3 {
  fill: black !important; }


#du9tpl1wbp9 .crit0,
#du9tpl1wbp9 .crit1,
#du9tpl1wbp9 .crit2,
#du9tpl1wbp9 .crit3 {
  stroke: #ff8888;
  fill: red;
  stroke-width: 2; }

#du9tpl1wbp9 .activeCrit0,
#du9tpl1wbp9 .activeCrit1,
#du9tpl1wbp9 .activeCrit2,
#du9tpl1wbp9 .activeCrit3 {
  stroke: #ff8888;
  fill: #bfc7ff;
  stroke-width: 2; }

#du9tpl1wbp9 .doneCrit0,
#du9tpl1wbp9 .doneCrit1,
#du9tpl1wbp9 .doneCrit2,
#du9tpl1wbp9 .doneCrit3 {
  stroke: #ff8888;
  fill: lightgrey;
  stroke-width: 2;
  cursor: pointer;
  shape-rendering: crispEdges; }

#du9tpl1wbp9 .doneCritText0,
#du9tpl1wbp9 .doneCritText1,
#du9tpl1wbp9 .doneCritText2,
#du9tpl1wbp9 .doneCritText3 {
  fill: black !important; }

#du9tpl1wbp9 .activeCritText0,
#du9tpl1wbp9 .activeCritText1,
#du9tpl1wbp9 .activeCritText2,
#du9tpl1wbp9 .activeCritText3 {
  fill: black !important; }

#du9tpl1wbp9 .titleText {
  text-anchor: middle;
  font-size: 18px;
  fill: black; }

#du9tpl1wbp9 g.classGroup text {
  fill: #9370DB;
  stroke: none;
  font-family: 'trebuchet ms', verdana, arial;
  font-size: 10px; }

#du9tpl1wbp9 g.classGroup rect {
  fill: #ECECFF;
  stroke: #9370DB; }

#du9tpl1wbp9 g.classGroup line {
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 .classLabel .box {
  stroke: none;
  stroke-width: 0;
  fill: #ECECFF;
  opacity: 0.5; }

#du9tpl1wbp9 .classLabel .label {
  fill: #9370DB;
  font-size: 10px; }

#du9tpl1wbp9 .relation {
  stroke: #9370DB;
  stroke-width: 1;
  fill: none; }

#du9tpl1wbp9 #compositionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 #compositionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 #aggregationStart {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 #aggregationEnd {
  fill: #ECECFF;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 #dependencyStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 #dependencyEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 #extensionStart {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 #extensionEnd {
  fill: #9370DB;
  stroke: #9370DB;
  stroke-width: 1; }

#du9tpl1wbp9 .commit-id,
#du9tpl1wbp9 .commit-msg,
#du9tpl1wbp9 .branch-label {
  fill: lightgrey;
  color: lightgrey; }



#du9tpl1wbp9 .label{
  color:#18B14E;
}
#du9tpl1wbp9 .te-md-container--dark .node rect {
  fill: red;
}

#du9tpl1wbp9 .node rect,
#du9tpl1wbp9 .node circle,
#du9tpl1wbp9 .node ellipse,
#du9tpl1wbp9 .node polygon {
  fill: #F9FFFB;;
  stroke: #2DBD60;
  stroke-width: 1.5px;
}
#du9tpl1wbp9 .arrowheadPath{
  fill: #2DBD60;
}
#du9tpl1wbp9 .edgePath .path {
  stroke: #2DBD60;
  stroke-width: 1px;
}
#du9tpl1wbp9 .edgeLabel {
  background-color: #fff;
}
#du9tpl1wbp9 .cluster rect {
  fill: #F9FFFB !important;
  stroke: #2DBD60 !important;
  stroke-width: 1px !important;
}

#du9tpl1wbp9 .cluster text {
  fill: #F9FFFB;
}

#du9tpl1wbp9 div.mermaidTooltip {
  background: #F9FFFB;
  border: 1px solid #2DBD60;
}


#du9tpl1wbp9 .actor {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#du9tpl1wbp9 text.actor {
  fill: #2DBD60;
  stroke: none;
}

#du9tpl1wbp9 .actor-line {
  stroke: #2DBD60;
}

#du9tpl1wbp9 .messageLine0 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#du9tpl1wbp9 .messageLine1 {
  stroke-width: 1.5;
  stroke-dasharray: '2 2';
  stroke: #2DBD60;
}

#du9tpl1wbp9 #arrowhead {
  fill: #2DBD60;
}

#du9tpl1wbp9 #crosshead path {
  fill: #2DBD60 !important;
  stroke: #2DBD60 !important;
}

#du9tpl1wbp9 .messageText {
  fill: #2DBD60;
  stroke: none;
}

#du9tpl1wbp9 .labelBox {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#du9tpl1wbp9 .labelText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#du9tpl1wbp9 .loopText {
  fill: #2DBD60;
  stroke: #2DBD60;
}

#du9tpl1wbp9 .loopLine {
  stroke-width: 2;
  stroke-dasharray: '2 2';
  marker-end: 'url(#arrowhead)';
  stroke: #2DBD60;
}

#du9tpl1wbp9 .note {
  stroke: #2DBD60;
  fill: #F9FFFB;
}

#du9tpl1wbp9 .noteText {
  fill: #2DBD60;
  stroke: #2DBD60;
}


#du9tpl1wbp9 .section{
  opacity:1;
}
#du9tpl1wbp9 .section0,#du9tpl1wbp9  .section2 {
  fill: #ECF7F0;
}

#du9tpl1wbp9 .section1,
#du9tpl1wbp9 .section3 {
  fill: #FFF;
}
#du9tpl1wbp9 .taskText0,
#du9tpl1wbp9 .taskText1,
#du9tpl1wbp9 .taskText2,
#du9tpl1wbp9 .taskText3 {
  fill: #fff;
}

#du9tpl1wbp9 .task0,
#du9tpl1wbp9 .task1,
#du9tpl1wbp9 .task2,
#du9tpl1wbp9 .task3 {
  fill: #2DBD60;
  stroke: #359F5A;
}
</style><style>#du9tpl1wbp9 {
    color: rgb(244, 244, 244);
    font: normal normal normal normal 14px/22.399999618530273px monospace;
  }</style><g transform="translate(-12, -12)"><g class="output"><g class="clusters"></g><g class="edgePaths"><g class="edgePath" style="opacity: 1;"><path class="path" d="M74.484375,38.140625L116.4765625,38.140625L158.46875,38.140625" marker-end="url(#arrowhead9700)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9700" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M195.453125,38.140625L248.515625,38.140625L301.578125,38.140625" marker-end="url(#arrowhead9701)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9701" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M338.5625,38.140625L391.625,38.140625L444.6875,38.140625" marker-end="url(#arrowhead9702)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9702" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g><g class="edgePath" style="opacity: 1;"><path class="path" d="M481.671875,38.140625L534.734375,38.140625L587.796875,38.140625" marker-end="url(#arrowhead9703)" style="stroke: #333; stroke-width: 1.5px;fill:none"></path><defs><marker id="arrowhead9703" viewBox="0 0 10 10" refX="9" refY="5" markerUnits="strokeWidth" markerWidth="8" markerHeight="6" orient="auto"><path d="M 0 0 L 10 5 L 0 10 z" class="arrowheadPath" style="stroke-width: 1px; stroke-dasharray: 1px, 0px;"></path></marker></defs></g></g><g class="edgeLabels"><g class="edgeLabel" transform="translate(116.4765625,38.140625)" style="opacity: 1;"><g transform="translate(-20.40625,-8.0078125)" class="label"><rect rx="0" ry="0" width="34.171875" height="16.359375" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">midle</tspan></text></g></g><g class="edgeLabel" transform="translate(248.515625,38.140625)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="56.4375" height="16.359375" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">schedlink</tspan></text></g></g><g class="edgeLabel" transform="translate(391.625,38.140625)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="56.4375" height="16.359375" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">schedlink</tspan></text></g></g><g class="edgeLabel" transform="translate(534.734375,38.140625)" style="opacity: 1;"><g transform="translate(-36.609375,-8.0078125)" class="label"><rect rx="0" ry="0" width="56.4375" height="16.359375" style="fill:#e8e8e8;"></rect><text><tspan xml:space="preserve" dy="1em" x="1">schedlink</tspan></text></g></g></g><g class="nodes"><g class="node" id="A" transform="translate(47.2421875,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-27.2421875" y="-18.140625" width="54.484375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-17.2421875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">sched</tspan></text></g></g></g><g class="node" id="B" transform="translate(176.9609375,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-18.4921875" y="-18.140625" width="36.984375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-8.4921875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">M1</tspan></text></g></g></g><g class="node" id="C" transform="translate(320.0703125,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-18.4921875" y="-18.140625" width="36.984375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-8.4921875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">M2</tspan></text></g></g></g><g class="node" id="D" transform="translate(463.1796875,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-18.4921875" y="-18.140625" width="36.984375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-8.4921875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">M3</tspan></text></g></g></g><g class="node" id="E" transform="translate(601.46875,38.140625)" style="opacity: 1;"><rect rx="0" ry="0" x="-13.671875" y="-18.140625" width="27.34375" height="36.28125"></rect><g class="label" transform="translate(0,0)"><g transform="translate(-3.671875,-8.140625)"><text><tspan xml:space="preserve" dy="1em" x="1">0</tspan></text></g></g></g></g></g></g></svg>
M的状态变迁

M在创建的时候会被设置成spinning状态。还有其他的spinning状态吗?
有:

  1. 当findrunnable准备进入stealing scheduler时,会先设置spinning
  2. 当findrunnable查找allp时,发现还有G,要重新回到循环来获取G

所以可以总结的是M的spinning是一种忙等的状态,而且它不是一个那么重要的状态,它可以看作一种系统的性能指标。如果spinning状态的M太多了,说明正常工作的M太少了。系统会组织M进入spinning状态。

// If number of spinning M's >= number of busy P's, block.
// This is necessary to prevent excessive CPU consumption
// when GOMAXPROCS>>1 but the program parallelism is low.
if !_g_.m.spinning && 2*atomic.Load(&sched.nmspinning) >= procs-atomic.Load(&sched.npidle) {
    goto stop
}

那么M什么时候离开spinning状态呢?有这几种情况:

  1. schedule中,M即将开始运行G,resetspinning将其spinning标志位设为false
  2. findrunnable没找到G,M即将调用stopm进入休眠状态

借用Golang 中 goroutine 的调度中的这张状态图:

这张图的状态变迁是正确的,但我觉得并不够贴切。因为对于M来说,没有很严格的状态变迁。只要知道M可以处于spinning状态,表示一种中间状态,在忙等待。而在正常工作和休眠的时候,都不是spinning。
M可以通过notesleep进入futexsleep状态,可以通过notewakeup唤醒休眠的M。

参考文献

%E6%9C%AC%E6%96%87%E5%9F%BA%E4%BA%8EGo%201.13%E3%80%82%0AGo%E6%8F%90%E4%BE%9B%E4%BA%86%E4%B8%80%E5%A5%97runtime%E6%9C%BA%E5%88%B6%EF%BC%8C%E6%9D%A5%E6%8F%90%E4%BE%9B%E5%85%B6%E5%8D%8F%E7%A8%8B%E8%B0%83%E5%BA%A6%E7%89%B9%E6%80%A7%E3%80%82%E5%AE%9E%E9%99%85%E4%B8%8A%E5%B0%B1%E6%98%AF%E5%AE%8C%E6%88%90%E4%B8%80%E7%A7%8D%E7%B1%BB%E4%BC%BCJavaScript%E7%9A%84event-driven%E7%BC%96%E7%A8%8B%E6%88%96%E8%80%85%E5%BC%82%E6%AD%A5%E7%BC%96%E7%A8%8B%E7%9A%84%E8%8C%83%E5%BC%8F%E3%80%82%E5%85%B6%E5%B8%A6%E6%9D%A5%E7%9A%84%E7%89%B9%E6%80%A7%E6%98%AF%EF%BC%8C%E5%A4%A7%E5%A4%A7%E5%A2%9E%E5%8A%A0%E4%BA%86Go%20app%E7%9A%84%E5%B9%B6%E5%8F%91%E7%89%B9%E6%80%A7%EF%BC%88%E5%8F%AF%E8%BE%BE%E7%99%BE%E4%B8%87%E7%BA%A7%EF%BC%8C%E7%94%9A%E8%87%B3%E6%9B%B4%E5%A4%9A%E7%9A%84%E5%B9%B6%E5%8F%91%EF%BC%89%E3%80%82%E7%89%B9%E5%88%AB%E9%80%82%E7%94%A8%E4%BA%8E%E6%89%80%E6%9C%89%E5%B9%B6%E5%8F%91%E9%83%BD%E6%98%AF%E6%97%A0%E5%B7%AE%E5%88%AB%E7%9A%84%E5%9C%BA%E6%99%AF%E4%B8%8B%E3%80%82%E4%BE%8B%E5%A6%82%E7%BD%91%E7%BB%9C%E8%BF%9E%E6%8E%A5%E3%80%82%E4%BD%86%E5%9B%A0%E4%B8%BA%E6%B2%A1%E6%9C%89%E4%BC%98%E5%85%88%E7%BA%A7%E6%A6%82%E5%BF%B5%EF%BC%8C%E5%AE%9E%E9%99%85%E4%B8%8AGo%E7%9A%84%E8%B0%83%E5%BA%A6%E6%A1%86%E6%9E%B6%E5%AF%B9%E5%AE%9E%E6%97%B6%E6%80%A7%E6%98%AF%E6%9C%89%E6%8D%9F%E5%AE%B3%E7%9A%84%E3%80%82%0A%E6%9C%AC%E6%96%87%E4%B8%93%E6%B3%A8%E4%BA%8E%E5%88%86%E6%9E%90%E8%B0%83%E5%BA%A6%E6%A1%86%E6%9E%B6%EF%BC%8C%E4%BB%A5%E5%8F%8A%E4%B8%80%E4%BA%9B%E6%A1%86%E6%9E%B6%E7%9A%84%E5%AE%9E%E7%8E%B0%E7%BB%86%E8%8A%82%E3%80%82%E6%96%87%E6%9C%AB%E4%BC%9A%E5%88%97%E4%B8%BE%E6%88%91%E5%9C%A8%E9%98%85%E8%AF%BB%E6%97%B6%E5%8F%82%E8%80%83%E7%9A%84%E7%BD%91%E9%A1%B5%E9%93%BE%E6%8E%A5%E3%80%82%0A%5Btoc%5D%0A%23%23%20GPM%0AGo%E5%9F%BA%E4%BA%8EGPM%E8%BF%99%E4%B8%89%E4%B8%AAobject%E6%9D%A5%E7%BB%84%E7%BB%87%E8%B0%83%E5%BA%A6%E6%A1%86%E6%9E%B6%E3%80%82%E5%A4%A7%E8%87%B4%E7%BB%93%E6%9E%84%E5%A6%82%E4%B8%8B%EF%BC%9A%0A!%5B11413e494428c2540e349637f34bbec3.png%5D(evernotecid%3A%2F%2F22617523-9521-4D00-B771-5F27B85F00EB%2Fappyinxiangcom%2F161681%2FENResource%2Fp6067)%0A%E5%85%B6%E4%B8%AD%EF%BC%9A%0A-%20G%3A%20Gourtines%2C%20%E6%AF%8F%E4%B8%AAGoroutine%E5%AF%B9%E5%BA%94%E4%B8%80%E4%B8%AAG%E7%BB%93%E6%9E%84%E4%BD%93%EF%BC%8CG%E4%BF%9D%E5%AD%98Goroutine%E7%9A%84%E8%BF%90%E8%A1%8C%E5%A0%86%E6%A0%88%EF%BC%8C%E5%8D%B3%E5%B9%B6%E5%8F%91%E4%BB%BB%E5%8A%A1%E7%8A%B6%E6%80%81%E3%80%82%0A-%20P%3A%20Processors%2C%20%E7%90%86%E6%83%B3%E7%8A%B6%E6%80%81%E4%B8%8B%E4%B8%80%E4%B8%AAP%E5%AF%B9%E5%BA%94%E4%B8%80%E4%B8%AACPU%E6%A0%B8%E3%80%82%E5%AF%B9M%E6%9D%A5%E8%AF%B4%EF%BC%8CP%E6%8F%90%E4%BE%9B%E4%BA%86%E7%9B%B8%E5%85%B3%E7%9A%84%E6%89%A7%E8%A1%8C%E7%8E%AF%E5%A2%83(Context)%EF%BC%8C%E5%A6%82%E5%86%85%E5%AD%98%E5%88%86%E9%85%8D%E7%8A%B6%E6%80%81(mcache)%EF%BC%8C%E4%BB%BB%E5%8A%A1%E9%98%9F%E5%88%97(G)%E7%AD%89%0A-%20M%3A%20Machine%2C%20OS%E7%BA%BF%E7%A8%8B%E6%8A%BD%E8%B1%A1%EF%BC%8C%E5%BD%93%E5%92%8C%E6%9F%90%E4%B8%AAP%E7%BB%91%E5%AE%9A%EF%BC%8C%E4%BB%8EP%E7%9A%84runq%E4%B8%AD%E4%B8%8D%E6%96%AD%E5%8F%96%E5%87%BAG%EF%BC%8C%E5%88%87%E6%8D%A2%E5%A0%86%E6%A0%88%E5%B9%B6%E6%89%A7%E8%A1%8C%EF%BC%8CM%E6%9C%AC%E8%BA%AB%E4%B8%8D%E5%85%B7%E5%A4%87%E6%89%A7%E8%A1%8C%E7%8A%B6%E6%80%81%EF%BC%8C%E5%9C%A8%E9%9C%80%E8%A6%81%E4%BB%BB%E5%8A%A1%E5%88%87%E6%8D%A2%E6%97%B6%EF%BC%8CM%E5%B0%86%E5%A0%86%E6%A0%88%E7%8A%B6%E6%80%81%E5%86%99%E5%9B%9EG%EF%BC%8C%E4%BB%BB%E4%BD%95%E5%85%B6%E5%AE%83M%E9%83%BD%E8%83%BD%E6%8D%AE%E6%AD%A4%E6%81%A2%E5%A4%8D%E6%89%A7%E8%A1%8C%E3%80%82%0A%23%23%20schedule%0A%E8%B0%83%E5%BA%A6%E5%A7%8B%E4%BA%8EGo%20bootstrap%E8%BF%87%E7%A8%8B%E7%9A%84mstart%EF%BC%8C%E6%AF%8F%E4%B8%80%E8%BD%AE%E8%B0%83%E5%BA%A6%EF%BC%8C%E5%AE%9E%E9%99%85%E5%B0%B1%E6%98%AF%E4%B8%80%E6%AC%A1schedule%E5%87%BD%E6%95%B0%E7%9A%84%E8%B0%83%E7%94%A8%E3%80%82%0A%E8%AF%A5%E5%87%BD%E6%95%B0%E5%AE%9A%E4%B9%89%E5%9C%A8go%2Fsrc%2Fruntime%2Fproc.go%E4%B8%AD%E3%80%82%0A%E6%9C%AC%E6%96%87%E9%92%88%E5%AF%B9%E8%B0%83%E5%BA%A6%E5%99%A8%E8%BF%9B%E8%A1%8C%E5%88%86%E6%9E%90%EF%BC%8C%E4%BB%8D%E7%84%B6%E5%BF%BD%E7%95%A5%E4%BB%A5%E4%B8%8B%E7%89%B9%E6%80%A7%EF%BC%8C%E5%8C%85%E6%8B%AC%E4%BD%86%E4%B8%8D%E9%99%90%E4%BA%8E%EF%BC%9A%0A-%20garbage%20collection%0A-%20C%20Go%0A-%20Linux%20signal%E5%A4%84%E7%90%86%0A-%20Trace%E6%93%8D%E4%BD%9C%0A-%20%E6%A0%88%E6%93%8D%E4%BD%9C%0A%E4%B8%80%E6%AC%A1%E8%B0%83%E5%BA%A6%E7%9A%84%E6%B5%81%E7%A8%8B%E5%A6%82%E4%B8%8B%EF%BC%9A%0A%60%60%60mermaid%0Agraph%20TD%0AA%5Bschedule%5D%20--%3E%20B%7Bschedtick%20%25%2061%20%3D%3D%200%20%3F%7D%0AB%20--%3E%20%7CN%7C%20D%0AB%20--%3E%20%7CY%7C%20C%5B%22globrunqget(_g_.m.p)%22%5D%0AC%20--%3E%7Cgp%20%3D%3D%20nil%7C%20D%5B%22runqget(_g_.m.p)%22%5D%0AD%20--%3E%7Cgp%20%3D%3D%20nil%7C%20E%5Bfindrunnable%5D%0AE%20--%3E%20G%7B_g_.m.spinning%20%3D%3D%201%20%3F%7D%0AG%20--%3E%20%7CY%7C%20H%5Bresetspinning%5D%0AH%20--%3E%20F%5B%22execute(gp)%22%5D%0AC%20--%3E%7Cgp%20!%3D%20nil%7C%20G%0AD%20--%3E%7Cgp%20!%3D%20nil%7C%20G%0AG%20--%3E%20%7CN%7C%20F%0A%60%60%60%0A%23%23%23%20globrunqget%20%26%20runqget%0A%E8%BF%99%E4%B8%A4%E4%B8%AA%E5%87%BD%E6%95%B0%E9%9D%9E%E5%B8%B8%E7%B1%BB%E4%BC%BC%EF%BC%8C%E4%B8%80%E4%B8%AA%E4%BB%8E%E5%85%A8%E5%B1%80%E9%98%9F%E5%88%97%E4%B8%AD%E5%8F%96%EF%BC%8C%E4%B8%80%E4%B8%AA%E4%BB%8E%E6%9C%AC%E5%9C%B0%E9%98%9F%E5%88%97%E4%B8%AD%E5%8F%96%E3%80%82%E5%8F%96%E7%9A%84%E6%97%B6%E6%9C%BA%E5%90%84%E6%9C%89%E4%B8%8D%E5%90%8C%E3%80%82%0A%23%23%23%23%20globrunqget%0AGo%E8%B0%83%E5%BA%A6%E5%99%A8%E4%B8%BA%E4%BA%86%E5%85%A8%E5%B1%80%E9%98%9F%E5%88%97%E4%B8%AD%E7%9A%84G%E4%B8%8D%E8%A2%AB%E9%A5%BF%E6%AD%BB%EF%BC%8C%E6%B2%A161%E6%AC%A1%E8%B0%83%E5%BA%A6%E9%83%BD%E4%BC%9A%E6%A3%80%E6%9F%A5%E4%B8%80%E4%B8%8B%E5%85%A8%E5%B1%80%E9%98%9F%E5%88%97%E4%B8%AD%E7%9A%84G%EF%BC%8C%E5%B9%B6%E5%8F%96%E5%87%BA%E8%BF%90%E8%A1%8C%E3%80%82%0A%60%60%60mermaid%0Agraph%20TD%0AA%5Bglobrunqget%5D%20--%3E%20%7C%22runqsize%20%3D%3D%200%22%7C%20H%5B%E8%BF%94%E5%9B%9Enil%5D%0AA%20--%3E%20%7C%22runqsize%20!%3D%200%22%7C%20B%5Bn%20%3A%3D%20sched.runqsize%2Fgomaxprocs%20%2B%201%5D%0AB%20--%3E%20C%5B%E8%AE%BE%E7%BD%AE%E5%90%88%E9%80%82%E7%9A%84n%E5%80%BC%5D%0AC%20--%3E%20D%5Brunqsize%20-%3D%20n%5D%0AD%20--%3E%20E%5B%22%E4%BB%8Erunq%E4%B8%ADpop%20G%EF%BC%8Cz%E4%BD%9C%E4%B8%BA%E8%BF%94%E5%9B%9E%E5%80%BC%22%5D%0AE%20--%3E%20F%5B%22%E4%BB%8Eglobal%20runq%E4%B8%AD%E5%8F%96%E5%87%BAn%E4%B8%AAG%EF%BC%8C%E6%94%BE%E5%85%A5P%E7%9A%84runq%22%5D%0AF%20--%3E%20G%5Breturn%20gp%5D%0A%60%60%60%0Aglobrunqget%E7%9A%84%E5%87%BD%E6%95%B0%E5%8E%9F%E5%9E%8B%E6%98%AF%EF%BC%9A%0A%60%60%60go%0Afunc%20globrunqget(_p_%20*p%2C%20max%20int32)%20*g%0A%60%60%60%0A%E4%B8%8A%E5%9B%BE%E4%B8%ADn%E6%98%AF%E8%A6%81%E5%8F%96%E5%87%BA%E7%9A%84G%E7%9A%84%E4%B8%AA%E6%95%B0%E3%80%82%E8%BF%99%E9%87%8C%E5%A6%82%E6%9E%9C%E8%80%83%E8%99%91max%E4%BC%A0%E5%85%A5%E5%A4%A7%E4%BA%8E0%E7%9A%84%E6%95%B0%EF%BC%8C%E5%88%99n%E5%8F%96%E5%80%BC%E9%9C%80%E8%A6%81%E8%80%83%E8%99%91%EF%BC%9A%0A1.%20%E4%B8%8D%E8%83%BD%E8%B6%85%E8%BF%87runqsize%EF%BC%8Crunqsize%E6%98%AF%E6%AD%A4%E6%97%B6global%20queue%E4%B8%ADG%E7%9A%84%E4%B8%AA%E6%95%B0%0A2.%20%E4%B8%8D%E8%83%BD%E8%B6%85%E8%BF%87%E4%BC%A0%E5%85%A5%E7%9A%84max%E5%80%BC%0A3.%20%E4%B8%8D%E8%83%BD%E8%B6%85%E8%BF%87%E4%B8%80%E4%B8%AAP%E7%9A%84runqsize%EF%BC%8C%E8%AF%A5size%E5%AE%9A%E4%B9%89%E5%9C%A8runtime2.go%E7%9A%84type%20p%E5%A3%B0%E6%98%8E%E4%B8%AD%2C%E5%BD%93%E5%89%8D%E5%80%BC%E4%B8%BA256%E3%80%82%0A%60%60%60go%0Atype%20p%20struct%20%7B%0A%20%20%20%20...%0A%20%20%20%20runq%20%20%20%20%20%5B256%5Dguintptr%0A%20%20%20%20...%0A%7D%0A%60%60%60%0A%E5%9C%A8schedule%E5%87%BD%E6%95%B0%E4%B8%ADglobrunqget%E5%87%BD%E6%95%B0%E7%9A%84max%E5%8F%82%E6%95%B0%E4%BC%A0%E5%85%A5%E7%9A%84%E6%98%AF0%EF%BC%8C%E6%89%80%E4%BB%A5%E4%B8%8A%E8%BF%B03%E4%B8%AA%E6%9D%A1%E4%BB%B6%E4%B8%AD%E7%9A%84%E6%9D%A1%E4%BB%B62%E5%B0%B1%E4%B8%8D%E7%94%A8%E8%80%83%E8%99%91%E4%BA%86%E3%80%82%0A%23%23%23%23%20runqget%0Arunqget%E4%BC%9A%E5%85%88%E6%9F%A5%E7%9C%8BP%E7%9A%84runnext%E6%88%90%E5%91%98%EF%BC%8C%E5%A6%82%E6%9E%9C%E6%9C%89%EF%BC%8C%E5%88%99%E8%BF%94%E5%9B%9Erunnext%E6%8C%87%E5%90%91%E7%9A%84G%0A%E5%90%A6%E5%88%99%E4%BC%9A%E5%8F%96%E5%BD%93%E5%89%8Drunqhead%E6%8C%87%E5%90%91%E7%9A%84runq%E6%95%B0%E7%BB%84%E7%9A%84%E6%88%90%E5%91%98G%E3%80%82%0Arunqhead%E5%92%8Crunqtail%E6%98%AF%E4%B8%A4%E4%B8%AAint%E7%B4%A2%E5%BC%95%E5%80%BC%E3%80%82%E5%A6%82%E6%9E%9C%E4%B8%A4%E5%80%BC%E7%9B%B8%E7%AD%89%E8%A1%A8%E7%A4%BA%E9%98%9F%E5%88%97%E6%98%AF%E7%A9%BA%E7%9A%84%E3%80%82%0A%0A%23%23%23%20findrunnable%0Afindrunnable%E6%98%AF%E4%B8%80%E4%B8%AA%E9%98%BB%E5%A1%9E%E5%9E%8B%E8%B0%83%E7%94%A8%EF%BC%8C%E6%9D%A5%E5%AF%BB%E6%89%BErunnable%20G%E3%80%82%E5%A6%82%E6%9E%9C%E6%89%BE%E4%B8%8D%E5%88%B0%EF%BC%8C%E5%B0%B1%E4%BC%9A%E8%B0%83%E7%94%A8stopm%EF%BC%8C%E6%9D%A5%E4%BC%91%E7%9C%A0M%E3%80%82%E5%85%B6%E5%9F%BA%E6%9C%AC%E6%B5%81%E7%A8%8B%E5%A6%82%E4%B8%8B%EF%BC%9A%0A%60%60%60mermaid%0Agraph%20TD%0AA%5Bfindrunnable%5D%20--%3E%20B%5B%22runqget(_p_)%22%5D%0AB%20--%3E%20%7Cno%20G%7C%20C%5B%22globrunqget(_p_%2C%20max%3D0)%22%5D%0AB%20--%3E%20%7Chas%20G%7C%20J%5Breturn%5D%0AC%20--%3E%20%7Cno%20G%7C%20D%5Bsteal%20G%20from%20other%20P%5D%0AD%20--%3E%20%7Cno%20G%7C%20E%5Bgoing%20to%20stop%5D%0AE%20--%3E%20F%5Bbefore%20stop%2C%20check%20globrunqget%5D%0AF%20--%3E%20G%5B%22pidleput(_p_)%22%5D%0AG%20--%3E%20H%5Bbefore%20stop%2C%20check%20all%20P%5D%0AH%20--%3E%20I%5Bstopm%5D%0AI%20--%3E%20%7Cwakeup%7C%20B%0AF%20--%3E%20%7Csched.runqsize%20!%3D%200%7C%20B%0AH%20--%3E%20%7C!runqempty%7C%20B%0A%0AC%20--%3E%20%7Chas%20G%7C%20J%0AD%20--%3E%20%7Chas%20G%7C%20J%0A%60%60%60%0A%E8%BF%99%E9%87%8C%E5%BF%BD%E7%95%A5%E4%BA%86net%20poller%EF%BC%8C%E8%BF%99%E6%98%AF%E7%94%A8%E6%9D%A5%E5%A4%84%E7%90%86%E5%BC%82%E6%AD%A5%E7%B3%BB%E7%BB%9F%E8%B0%83%E7%94%A8%E7%9A%84%E4%B8%80%E7%A7%8D%E6%96%B9%E5%BC%8F%E3%80%82%0A%E4%BB%A5%E4%B8%8A%E6%B5%81%E7%A8%8B%E4%B8%AD%E9%99%A4%E4%BA%86%E5%B8%B8%E8%A7%84%E7%9A%84global%20q%E5%92%8Clocal%20q%E6%93%8D%E4%BD%9C%EF%BC%8C%E8%BF%98%E6%9C%89%E4%B8%80%E6%AE%B5%E5%85%B3%E4%BA%8Estealing%20G%E7%9A%84%E7%AE%97%E6%B3%95%EF%BC%8C%E7%A7%B0%E4%B8%BAGo%20Stealing%20Scheduler%EF%BC%8C%E5%8F%AF%E4%BB%A5%E5%8F%82%E8%80%83%5BGo's%20work-stealing%20scheduler%5D(https%3A%2F%2Frakyll.org%2Fscheduler%2F)%E3%80%82%E5%85%B7%E4%BD%93%E4%BB%A3%E7%A0%81%E5%A6%82%E4%B8%8B%EF%BC%9A%0A%60%60%60go%0Afor%20i%20%3A%3D%200%3B%20i%20%3C%204%3B%20i%2B%2B%20%7B%0A%20%20%20%20for%20enum%20%3A%3D%20stealOrder.start(fastrand())%3B%20!enum.done()%3B%20enum.next()%20%7B%0A%20%20%20%20%20%20%20%20if%20sched.gcwaiting%20!%3D%200%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20goto%20top%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%20%20%20%20stealRunNextG%20%3A%3D%20i%20%3E%202%20%2F%2F%20first%20look%20for%20ready%20queues%20with%20more%20than%201%20g%0A%20%20%20%20%20%20%20%20if%20gp%20%3A%3D%20runqsteal(_p_%2C%20allp%5Benum.position()%5D%2C%20stealRunNextG)%3B%20gp%20!%3D%20nil%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20return%20gp%2C%20false%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%7D%0A%7D%0A%60%60%60%0A%E8%AF%B4%E5%AE%9E%E8%AF%9D%E4%BB%A3%E7%A0%81%E6%B2%A1%E6%9C%89%E5%BE%88%E7%9C%8B%E6%87%82%EF%BC%9A%0A-%20%E8%BF%99%E4%B8%AAi%20%3D%20%5B0..4%5D%E6%98%AF%E6%80%8E%E4%B9%88%E5%9B%9E%E4%BA%8B%E5%84%BF%EF%BC%9F%0A-%20%E7%AC%AC%E4%BA%8C%E4%B8%AAfor%E5%BE%AA%E7%8E%AF%E4%BC%BC%E4%B9%8E%E9%87%87%E7%94%A8%E9%9A%8F%E6%9C%BA%E6%95%B0%E6%9D%A5%E9%9A%8F%E6%9C%BA%E8%8E%B7%E5%BE%97%E7%9B%B8%E5%BA%94%E7%9A%84P%EF%BC%8C%E6%9D%A5steal%20G%EF%BC%8C%E4%BD%86%E6%98%AF%E4%B8%BA%E5%95%A5%E6%9C%89%E4%B8%AA%60stealRunNextG%20%3A%3D%20i%20%3E%202%60%0A%0AstealRunNextG%E6%A0%B9%E6%8D%AE%E4%BB%A3%E7%A0%81%E7%90%86%E8%A7%A3%EF%BC%8C%E5%BA%94%E8%AF%A5%E6%98%AFbool%E5%8F%98%E9%87%8F%EF%BC%8C%E8%A1%A8%E7%A4%BA%E6%98%AF%E5%90%A6%E4%BB%8EP%E7%9A%84runnext%E6%88%90%E5%91%98%E8%8E%B7%E5%8F%96G%E3%80%82%0Arunqsteal%E4%BC%9A%E4%BB%8Erunnext%E5%81%B7%E5%8F%96G%EF%BC%8C%E6%88%96%E8%80%85%E4%BC%9A%E5%81%B7%E5%8F%96P%E7%9A%84runq%E4%B8%AD%E4%B8%80%E5%8D%8A%E7%9A%84G%0A%0A%23%23%23%20resetspinning%0A%E8%BF%99%E4%B8%AA%E5%87%BD%E6%95%B0%E5%B0%86%E5%8E%9F%E5%85%88%E4%B8%BAtrue%E7%9A%84_g_.m.spinning%E8%AE%BE%E4%B8%BAfalse%E3%80%82%E5%BD%93%E5%85%A8%E5%B1%80%E9%83%BD%E6%B2%A1%E6%9C%89spinning%E7%9A%84M%E4%BA%86%EF%BC%8C%E5%88%99%E8%A6%81%E5%94%A4%E9%86%92idle%20P%E3%80%82%E6%B2%A1%E6%9C%89spinning%20M%E8%A1%A8%E7%A4%BA%E6%89%80%E6%9C%89%E7%9A%84M%E8%A6%81%E4%B9%88%E5%9C%A8%E5%B7%A5%E4%BD%9C%EF%BC%88%E5%A4%84%E7%90%86G%EF%BC%89%EF%BC%8C%E8%A6%81%E4%B9%88%E5%9C%A8%E4%BC%91%E7%9C%A0%E3%80%82%E5%A6%82%E6%9E%9C%E6%AD%A4%E6%97%B6%E6%9C%89idle%20P%EF%BC%8C%E8%A1%A8%E7%A4%BA%E6%9C%89CPU%E7%A9%BA%E9%97%B2%EF%BC%8C%E5%88%99%E6%AD%A4%E6%97%B6%E6%98%AF%E4%B8%80%E4%B8%AA%E5%A5%BD%E7%9A%84%E6%97%B6%E6%9C%BA%E6%9D%A5%E5%94%A4%E9%86%92M%E5%A4%84%E7%90%86%E6%9B%B4%E5%A4%9A%E7%9A%84%E5%B7%A5%E4%BD%9C%E3%80%82%0A%E5%AE%9E%E9%99%85%E4%B8%8AGPM%E4%B8%89%E8%80%85%E9%83%BD%E6%9C%89%E8%87%AA%E5%B7%B1%E7%9A%84%E7%8A%B6%E6%80%81%E6%9C%BA%EF%BC%8CG%E5%92%8CP%E6%AF%94%E8%BE%83%E6%B8%85%E6%99%B0%EF%BC%8CM%E7%9A%84%E7%8A%B6%E6%80%81%E5%B0%B1%E6%AF%94%E8%BE%83%E6%A8%A1%E7%B3%8A%EF%BC%8C%E5%88%B0%E5%BA%95M%E6%9C%89%E5%87%A0%E7%A7%8D%E7%8A%B6%E6%80%81%EF%BC%8C%E7%89%B9%E5%88%AB%E7%9A%84%E6%98%AFM%E4%BC%9A%E5%9C%A8%E4%BD%95%E6%97%B6spinning%E3%80%82%E5%90%8E%E6%96%87%E4%BC%9A%E4%BB%8B%E7%BB%8D%E3%80%82%0A%0A%23%23%23%20execute%0A%E7%AE%80%E5%8D%95%E6%9D%A5%E7%9C%8B%EF%BC%8Cexecute%E5%8F%AA%E5%81%9A%E4%BA%86%E4%B8%A4%E4%BB%B6%E4%BA%8B%EF%BC%9A%0A1.%20%E5%88%87%E6%8D%A2G%20status%EF%BC%9A_Grunnalbe%20--%3E%20%5C_Grunning%0A2.%20%E8%B0%83%E7%94%A8%60gogo(%26gp.sched)%60%0A%0AARM%E5%B9%B3%E5%8F%B0%E7%9A%84gogo%E5%87%BD%E6%95%B0%E5%AE%9E%E7%8E%B0%E5%9C%A8%EF%BC%9Ago%2Fsrc%2Fruntime%2Fasm_arm.s%EF%BC%8C%E6%98%AF%E4%B8%80%E4%B8%AA%E6%B1%87%E7%BC%96%E5%87%BD%E6%95%B0%E3%80%82%E4%B8%BB%E8%A6%81%E5%81%9A%E4%BA%86%E4%BB%A5%E4%B8%8B%E5%87%A0%E4%BB%B6%E4%BA%8B%E6%83%85%EF%BC%9A%0A-%20%E8%AE%BE%E7%BD%AETLS%E4%B8%AD%E7%9A%84G%0A-%20%E5%81%9Acontext%20switch%0A-%20%E8%B7%B3%E8%BD%AC%E5%88%B0%E5%AF%B9%E5%BA%94%E7%9A%84PC%E5%8F%96%E8%BF%90%E8%A1%8C%0A%0A%23%23%23%20GPM%E7%9A%84%E7%8A%B6%E6%80%81%E5%8F%98%E8%BF%81%0AGPM%E7%9A%84%E7%8A%B6%E6%80%81%E5%8F%98%E8%BF%81%E5%8F%AF%E4%BB%A5%E5%8F%82%E8%80%83%5BGolang%20%E4%B8%AD%20goroutine%20%E7%9A%84%E8%B0%83%E5%BA%A6%5D(https%3A%2F%2Fwww.ctolib.com%2Ftopics-133848.html)%E4%B8%80%E6%96%87%E4%B8%AD%E5%85%B3%E4%BA%8EGPM%E7%9A%84%E4%B8%80%E7%94%9F%E7%9A%84%E7%AB%A0%E8%8A%82%E3%80%82%0A%23%23%23%23%20G%20%26%20P%0A%E5%AF%B9G%E5%92%8CP%E4%B8%8D%E5%81%9A%E8%BF%87%E5%A4%9A%E7%9A%84%E8%A7%A3%E9%87%8A%EF%BC%8C%E7%8A%B6%E6%80%81%E6%9C%BA%E8%BF%98%E6%98%AF%E5%BE%88%E6%B8%85%E6%99%B0%E7%9A%84%E3%80%82%0AG%E7%94%B1newproc%E5%87%BD%E6%95%B0%E5%88%9B%E5%BB%BA%EF%BC%8C%E9%81%B5%E5%BE%AA%E4%B8%8B%E9%9D%A2%E7%9A%84%E7%8A%B6%E6%80%81%E5%8F%98%E5%8C%96%0A!%5Bf2c48d2f55f7311e173548cc09458871.png%5D(evernotecid%3A%2F%2F22617523-9521-4D00-B771-5F27B85F00EB%2Fappyinxiangcom%2F161681%2FENResource%2Fp6068)%0A%0AP%E7%94%B1bootstrap%E8%BF%87%E7%A8%8B%E7%9A%84procresize%E5%88%9B%E5%BB%BA%EF%BC%8C%E9%81%B5%E5%BE%AA%E4%B8%8B%E9%9D%A2%E7%9A%84%E7%8A%B6%E6%80%81%E6%9C%BA%E5%8F%98%E8%BF%81%EF%BC%9A%0A!%5B9e80b53de7666672eb5ce4e9174cdff6.png%5D(evernotecid%3A%2F%2F22617523-9521-4D00-B771-5F27B85F00EB%2Fappyinxiangcom%2F161681%2FENResource%2Fp6069)%0A%0A%23%23%23%23%20M%E7%9A%84%E7%8A%B6%E6%80%81%0A%E8%BF%99%E9%87%8C%E4%B8%8D%E8%AE%A8%E8%AE%BA%E7%89%B9%E6%AE%8A%E7%9A%84M%EF%BC%8C%E4%BE%8B%E5%A6%82%EF%BC%9Asysmon%E6%88%96%E8%80%85finalizer%0A%E6%99%AE%E9%80%9A%E7%9A%84M%E7%94%B1startm%E5%87%BD%E6%95%B0%E5%88%9B%E5%BB%BA%E3%80%82%E5%85%B6%E5%A4%A7%E8%87%B4%E6%B5%81%E7%A8%8B%E5%A6%82%E4%B8%8B%EF%BC%9A%0A%60%60%60mermaid%0Agraph%20TD%0AA%5B%22startm(_p_%2C%20spinning)%22%5D%20--%3E%7C_p_%20%3D%3D%20nil%7C%20B%5Bpidleget%5D%0AB%20--%3E%20%7Cfind%20idle%20P%7C%20C%5Bmget%5D%0AA%20--%3E%20C%0AB%20--%3E%20%7Cno%20idle%20P%7C%20G%5Breturn%5D%0AC%20--%3E%20%7Cmp%20%3D%3D%20nil%7C%20D%5Bnewm%5D%0AD%20--%3E%20G%0AC%20--%3E%20E%5B%22mp.nextp.set(_p_)%22%5D%0AE%20--%3E%20F%5B%22notewakeup(%26mp.park)%22%5D%0AF%20--%3E%20G%0A%60%60%60%0A%0A%23%23%23%23%23%20M%E7%9A%84%E5%88%9B%E5%BB%BA%0A%E8%BF%99%E9%87%8C%E8%B7%9FM%E7%8A%B6%E6%80%81%E6%9C%89%E5%85%B3%E7%9A%84%E5%B0%B1%E6%98%AFM%E7%9A%84%E7%BA%BF%E7%A8%8B%E5%87%BD%E6%95%B0%2C%E4%BD%86%E8%AF%B7%E6%B3%A8%E6%84%8F%E4%B8%8B%E9%9D%A2%E8%BF%99%E4%B8%AAfn%E5%B9%B6%E4%B8%8D%E6%98%AF%E7%BA%BF%E7%A8%8B%E5%87%BD%E6%95%B0%E3%80%82%0A%60%60%60go%0Aif%20spinning%20%7B%0A%20%20%20%20%2F%2F%20The%20caller%20incremented%20nmspinning%2C%20so%20set%20m.spinning%20in%20the%20new%20M.%0A%20%20%20%20fn%20%3D%20mspinning%0A%7D%0A%60%60%60%0A%E7%9C%9F%E6%AD%A3%E7%9A%84%E7%BA%BF%E7%A8%8B%E5%87%BD%E6%95%B0%E5%9C%A8newosproc%E8%B0%83%E7%94%A8clone%E6%8E%A5%E5%8F%A3%E7%9A%84%E6%97%B6%E5%80%99%E4%BC%A0%E5%85%A5%E3%80%82Linux%E5%B9%B3%E5%8F%B0%E7%9A%84%E5%AE%9E%E7%8E%B0%E5%9C%A8os_linux.go%E4%B8%AD%E3%80%82%E5%A6%82%E4%B8%8B%EF%BC%9A%0A%60%60%60go%0Afunc%20clone(flags%20int32%2C%20stk%2C%20mp%2C%20gp%2C%20fn%20unsafe.Pointer)%20int32%0A%0Afunc%20newosproc(mp%20*m)%20%7B%0A%20%20%20%20...%0A%09ret%20%3A%3D%20clone(cloneFlags%2C%20stk%2C%20unsafe.Pointer(mp)%2C%20unsafe.Pointer(mp.g0)%2C%20unsafe.Pointer(funcPC(mstart)))%0A%20%20%20%20...%0A%7D%0A%60%60%60%0A%E6%89%80%E4%BB%A5M%E7%9A%84%E7%BA%BF%E7%A8%8B%E5%87%BD%E6%95%B0%E5%B0%B1%E6%98%AF%E6%9B%BE%E5%9C%A8bootstrap%E4%B8%AD%E5%87%BA%E7%8E%B0%E8%BF%87%EF%BC%8C%E8%BF%90%E8%A1%8CM0%E7%9A%84mstart%E5%87%BD%E6%95%B0%E3%80%82%E4%B8%8A%E9%9D%A2%E7%9A%84mspinning%E5%87%BD%E6%95%B0%E5%9C%A8mstart1%E4%B8%AD%E8%A2%AB%E8%B0%83%E7%94%A8%EF%BC%8C%E5%AE%83%E5%B0%86%E7%BA%BF%E7%A8%8B%E8%AE%BE%E7%BD%AE%E4%B8%BAspinning%E7%8A%B6%E6%80%81%E3%80%82%0A%0A%23%23%23%23%23%20idle%20M%E7%9A%84%E7%AE%A1%E7%90%86%0Amget%E4%BC%9A%E5%8E%BB%E8%8E%B7%E5%8F%96%E9%97%B2%E7%BD%AE%E7%9A%84M%E3%80%82%E8%80%8C%E6%89%80%E6%9C%89%E7%9A%84idle%20M%E5%B9%B6%E6%B2%A1%E6%9C%89%E4%B8%80%E4%B8%AA%E7%BB%9F%E4%B8%80%E7%9A%84%E4%BF%9D%E5%AD%98%E4%BD%8D%E7%BD%AE%EF%BC%8C%E8%80%8C%E6%98%AF%E9%80%9A%E8%BF%87M.schedlink%E5%BD%A2%E6%88%90%E4%B8%80%E4%B8%AA%E9%9A%90%E5%BD%A2%E7%9A%84%E9%93%BE%E8%A1%A8%E3%80%82%E7%9C%8Bmput%EF%BC%9A%0A%60%60%60go%0Afunc%20mput(mp%20*m)%20%7B%0A%09mp.schedlink%20%3D%20sched.midle%0A%09sched.midle.set(mp)%0A%09sched.nmidle%2B%2B%0A%20%20%20%20...%0A%7D%0A%60%60%60%0A%E5%81%87%E8%AE%BE%E8%B0%83%E7%94%A8%E4%BA%863%E6%AC%A1mput%EF%BC%8C%E9%82%A3%E4%B9%88%E6%AD%A4%E6%97%B6%E7%9A%84idle%E9%93%BE%E8%A1%A8%E5%BA%94%E8%AF%A5%E6%98%AF%E8%BF%99%E6%A0%B7%E7%9A%84%EF%BC%9A%0A%60%60%60mermaid%0Agraph%20LR%0AA%5Bsched%5D%20--%3E%7Cmidle%7C%20B%5BM1%5D%0AB%20--%3E%20%7Cschedlink%7C%20C%5BM2%5D%0AC%20--%3E%20%7Cschedlink%7C%20D%5BM3%5D%0AD%20--%3E%20%7Cschedlink%7C%20E%5B0%5D%0A%60%60%60%0A%0A%23%23%23%23%23%20M%E7%9A%84%E7%8A%B6%E6%80%81%E5%8F%98%E8%BF%81%0AM%E5%9C%A8%E5%88%9B%E5%BB%BA%E7%9A%84%E6%97%B6%E5%80%99%E4%BC%9A%E8%A2%AB%E8%AE%BE%E7%BD%AE%E6%88%90spinning%E7%8A%B6%E6%80%81%E3%80%82%E8%BF%98%E6%9C%89%E5%85%B6%E4%BB%96%E7%9A%84spinning%E7%8A%B6%E6%80%81%E5%90%97%EF%BC%9F%0A%E6%9C%89%EF%BC%9A%0A1.%20%E5%BD%93findrunnable%E5%87%86%E5%A4%87%E8%BF%9B%E5%85%A5stealing%20scheduler%E6%97%B6%EF%BC%8C%E4%BC%9A%E5%85%88%E8%AE%BE%E7%BD%AEspinning%0A2.%20%E5%BD%93findrunnable%E6%9F%A5%E6%89%BEallp%E6%97%B6%EF%BC%8C%E5%8F%91%E7%8E%B0%E8%BF%98%E6%9C%89G%EF%BC%8C%E8%A6%81%E9%87%8D%E6%96%B0%E5%9B%9E%E5%88%B0%E5%BE%AA%E7%8E%AF%E6%9D%A5%E8%8E%B7%E5%8F%96G%0A%0A%E6%89%80%E4%BB%A5%E5%8F%AF%E4%BB%A5%E6%80%BB%E7%BB%93%E7%9A%84%E6%98%AFM%E7%9A%84spinning%E6%98%AF%E4%B8%80%E7%A7%8D%E5%BF%99%E7%AD%89%E7%9A%84%E7%8A%B6%E6%80%81%EF%BC%8C%E8%80%8C%E4%B8%94%E5%AE%83%E4%B8%8D%E6%98%AF%E4%B8%80%E4%B8%AA%E9%82%A3%E4%B9%88%E9%87%8D%E8%A6%81%E7%9A%84%E7%8A%B6%E6%80%81%EF%BC%8C%E5%AE%83%E5%8F%AF%E4%BB%A5%E7%9C%8B%E4%BD%9C%E4%B8%80%E7%A7%8D%E7%B3%BB%E7%BB%9F%E7%9A%84%E6%80%A7%E8%83%BD%E6%8C%87%E6%A0%87%E3%80%82%E5%A6%82%E6%9E%9Cspinning%E7%8A%B6%E6%80%81%E7%9A%84M%E5%A4%AA%E5%A4%9A%E4%BA%86%EF%BC%8C%E8%AF%B4%E6%98%8E%E6%AD%A3%E5%B8%B8%E5%B7%A5%E4%BD%9C%E7%9A%84M%E5%A4%AA%E5%B0%91%E4%BA%86%E3%80%82%E7%B3%BB%E7%BB%9F%E4%BC%9A%E7%BB%84%E7%BB%87M%E8%BF%9B%E5%85%A5spinning%E7%8A%B6%E6%80%81%E3%80%82%0A%60%60%60go%0A%2F%2F%20If%20number%20of%20spinning%20M's%20%3E%3D%20number%20of%20busy%20P's%2C%20block.%0A%2F%2F%20This%20is%20necessary%20to%20prevent%20excessive%20CPU%20consumption%0A%2F%2F%20when%20GOMAXPROCS%3E%3E1%20but%20the%20program%20parallelism%20is%20low.%0Aif%20!_g_.m.spinning%20%26%26%202*atomic.Load(%26sched.nmspinning)%20%3E%3D%20procs-atomic.Load(%26sched.npidle)%20%7B%0A%20%20%20%20goto%20stop%0A%7D%0A%60%60%60%0A%0A%E9%82%A3%E4%B9%88M%E4%BB%80%E4%B9%88%E6%97%B6%E5%80%99%E7%A6%BB%E5%BC%80spinning%E7%8A%B6%E6%80%81%E5%91%A2%EF%BC%9F%E6%9C%89%E8%BF%99%E5%87%A0%E7%A7%8D%E6%83%85%E5%86%B5%EF%BC%9A%0A1.%20schedule%E4%B8%AD%EF%BC%8CM%E5%8D%B3%E5%B0%86%E5%BC%80%E5%A7%8B%E8%BF%90%E8%A1%8CG%EF%BC%8Cresetspinning%E5%B0%86%E5%85%B6spinning%E6%A0%87%E5%BF%97%E4%BD%8D%E8%AE%BE%E4%B8%BAfalse%0A2.%20findrunnable%E6%B2%A1%E6%89%BE%E5%88%B0G%EF%BC%8CM%E5%8D%B3%E5%B0%86%E8%B0%83%E7%94%A8stopm%E8%BF%9B%E5%85%A5%E4%BC%91%E7%9C%A0%E7%8A%B6%E6%80%81%0A%0A%E5%80%9F%E7%94%A8%5BGolang%20%E4%B8%AD%20goroutine%20%E7%9A%84%E8%B0%83%E5%BA%A6%5D(https%3A%2F%2Fwww.ctolib.com%2Ftopics-133848.html)%E4%B8%AD%E7%9A%84%E8%BF%99%E5%BC%A0%E7%8A%B6%E6%80%81%E5%9B%BE%EF%BC%9A%0A!%5B55d559f0bfd46d4bfecb35a969213c1c.png%5D(evernotecid%3A%2F%2F22617523-9521-4D00-B771-5F27B85F00EB%2Fappyinxiangcom%2F161681%2FENResource%2Fp6070)%0A%0A%E8%BF%99%E5%BC%A0%E5%9B%BE%E7%9A%84%E7%8A%B6%E6%80%81%E5%8F%98%E8%BF%81%E6%98%AF%E6%AD%A3%E7%A1%AE%E7%9A%84%EF%BC%8C%E4%BD%86%E6%88%91%E8%A7%89%E5%BE%97%E5%B9%B6%E4%B8%8D%E5%A4%9F%E8%B4%B4%E5%88%87%E3%80%82%E5%9B%A0%E4%B8%BA%E5%AF%B9%E4%BA%8EM%E6%9D%A5%E8%AF%B4%EF%BC%8C%E6%B2%A1%E6%9C%89%E5%BE%88%E4%B8%A5%E6%A0%BC%E7%9A%84%E7%8A%B6%E6%80%81%E5%8F%98%E8%BF%81%E3%80%82%E5%8F%AA%E8%A6%81%E7%9F%A5%E9%81%93M%E5%8F%AF%E4%BB%A5%E5%A4%84%E4%BA%8Espinning%E7%8A%B6%E6%80%81%EF%BC%8C%E8%A1%A8%E7%A4%BA%E4%B8%80%E7%A7%8D%E4%B8%AD%E9%97%B4%E7%8A%B6%E6%80%81%EF%BC%8C%E5%9C%A8%E5%BF%99%E7%AD%89%E5%BE%85%E3%80%82%E8%80%8C%E5%9C%A8%E6%AD%A3%E5%B8%B8%E5%B7%A5%E4%BD%9C%E5%92%8C%E4%BC%91%E7%9C%A0%E7%9A%84%E6%97%B6%E5%80%99%EF%BC%8C%E9%83%BD%E4%B8%8D%E6%98%AFspinning%E3%80%82%0AM%E5%8F%AF%E4%BB%A5%E9%80%9A%E8%BF%87notesleep%E8%BF%9B%E5%85%A5futexsleep%E7%8A%B6%E6%80%81%EF%BC%8C%E5%8F%AF%E4%BB%A5%E9%80%9A%E8%BF%87notewakeup%E5%94%A4%E9%86%92%E4%BC%91%E7%9C%A0%E7%9A%84M%E3%80%82%0A%0A%23%23%20%E5%8F%82%E8%80%83%E6%96%87%E7%8C%AE%0A-%20%5BScheduling%20In%20Go%20%3A%20Part%20II%20-%20Go%20Scheduler%5D(https%3A%2F%2Fwww.ardanlabs.com%2Fblog%2F2018%2F08%2Fscheduling-in-go-part2.html)%0A-%20%5BGo%20%E8%B0%83%E5%BA%A6%E6%A8%A1%E5%9E%8B%5D(https%3A%2F%2Fwudaijun.com%2F2018%2F01%2Fgo-scheduler%2F)%0A-%20%5BGo's%20work-stealing%20scheduler%5D(https%3A%2F%2Frakyll.org%2Fscheduler%2F)%0A-%20%5BGolang%20%E4%B8%AD%20goroutine%20%E7%9A%84%E8%B0%83%E5%BA%A6%5D(https%3A%2F%2Fwww.ctolib.com%2Ftopics-133848.html)