Skip to content

Latest commit

 

History

History
742 lines (686 loc) · 27 KB

webnn-status.md

File metadata and controls

742 lines (686 loc) · 27 KB
layout title permalink
page
Implementation Status of WebNN Operations
/webnn-status/
<style> .post img { height: 24px; margin: 4px 0; transform: none; display: inline-block; min-width: 24px; } .post .onnxrt img { width: 80px; } .post .tflite img { width: 102px; } .post table { font-size: 0.8rem; } .post table tr td { text-align: left; } .post table tr td:nth-child(1), .post table tr td:nth-child(2), .post table tr td:nth-child(4), .post table tr td:nth-child(6), .post table tr td:nth-child(8), .post table tr td:nth-child(10) { text-align: center; } .post table.framework tr td:nth-child(2), .post table.framework tr td:nth-child(4) { text-align: left; } .post table.framework tr td:nth-child(3), .post table.framework tr td:nth-child(5) { text-align: center; } .post table td { padding: 4px 4px; vertical-align: middle; } .post table th { padding: 4px 4px; } .impl_status { text-align: center; display: grid; grid-template-columns: 1fr 1fr 1fr; gap: 0px; border: 1px solid #dfe2e5; padding: 12px; font-size: 0.9em; } .impl_status_framework { text-align: center; display: grid; grid-template-columns: 1fr 1fr; gap: 0px; border: 1px solid #dfe2e5; padding: 12px; font-size: 0.9em; } .impl_status .title { font-size: 1rem; line-height: 1rem; } .post-content table th { font-weight: 400; font-size: 0.9rem; line-height: 0.9rem; } .post svg#w3 { transform: none; height: 36px; width: 72px; } sup { font-size: 0.7rem; } .impl_status .title { margin-bottom: 0.4rem; } .impl_status div { margin: 0 12px; } #cros { width: 6.4rem; height: 3.2rem; margin-left: 2px; margin-bottom: -3px; transform: rotateY(0deg) rotate(0deg); } .mt-2 { margin-top: 20px; } .tfop { width: 200px; } .tile { display: flex; flex-direction: row; justify-content: space-evenly; } .tile div { display: flex; flex-direction: column; align-items: center; } </style>
<style type="text/css"> .st0 { fill: #005A9C; }
          .st1 {
            fill: #221B0A;
          }
        </style>
        <g>
          <path class="st0" d="M118.4,76.6l24.2,82.4l24.2-82.4h17.5l-40.1,135.3h-1.7l-25.1-83.8l-25.1,83.8h-1.7L50.8,76.6h17.5L92.5,159
                l16.4-55.5l-8-26.9L118.4,76.6L118.4,76.6z" />
          <path class="st0" d="M234.2,168.5c0,12.3-3.3,22.6-9.8,30.9c-6.5,8.3-15,12.5-25.3,12.5c-7.8,0-14.6-2.5-20.4-7.4
                c-5.8-5-10.1-11.7-12.9-20.1l13.7-5.7c2,5.1,4.7,9.2,7.9,12.1c3.3,2.9,7.1,4.4,11.6,4.4c4.7,0,8.6-2.6,11.9-7.9
                c3.2-5.2,4.8-11.5,4.8-18.9c0-8.1-1.7-14.4-5.2-18.9c-4-5.2-10.3-7.9-18.9-7.9h-6.7v-8l23.4-40.4h-28.2l-7.9,13.4h-5V76.6h65.1v8.2
                l-24.7,42.6c8.7,2.8,15.3,7.9,19.7,15.2C232,149.9,234.2,158.6,234.2,168.5L234.2,168.5z" />
          <path class="st1"
            d="M302.2,75.9l2.8,17.3l-10.1,19.2c0,0-3.9-8.2-10.3-12.7c-5.4-3.8-8.9-4.6-14.4-3.5
                c-7.1,1.5-15.1,9.9-18.6,20.3c-4.2,12.5-4.2,18.5-4.4,24c-0.2,8.9,1.2,14.1,1.2,14.1s-6.1-11.3-6-27.8c0-11.8,1.9-22.5,7.4-33.1
                c4.8-9.3,11.9-14.9,18.3-15.5c6.6-0.7,11.7,2.5,15.7,5.9c4.2,3.6,8.5,11.4,8.5,11.4L302.2,75.9L302.2,75.9z" />
          <path class="st1" d="M303.4,173.6c0,0-4.4,7.9-7.2,11c-2.8,3.1-7.7,8.5-13.8,11.2c-6.1,2.7-9.3,3.2-15.4,2.6
                c-6-0.6-11.7-4.1-13.6-5.5c-2-1.4-7-5.8-9.8-9.7c-2.9-4-7.3-12-7.3-12s2.5,8,4,11.4c0.9,2,3.6,8,7.5,13.2
                c3.6,4.9,10.7,13.3,21.4,15.1c10.7,1.9,18.1-2.9,19.9-4.1c1.8-1.2,5.7-4.4,8.1-7c2.5-2.7,4.9-6.2,6.3-8.2c1-1.5,2.6-4.6,2.6-4.6
                L303.4,173.6L303.4,173.6z" />
        </g>
      </svg>
      <br />WebNN Spec
    </a>
  </th>
  <th rowspan="4">Web Platform<br />Tests</th>
  <th colspan="6">
    Chromium Implementation
  </th>
</tr>
    <tr class="title">
  <th colspan="2">
    LiteRT
    <sup><a href="#note-1">1</a></sup>
  </th>
  <th colspan="2">
    DirectML
    <sup><a href="#note-2">2</a></sup>
  </th>
     <th colspan="2">
      Core ML
      <sup><a href="#note-3">3</a></sup>
  </th>
</tr>
<tr class="title">
  <th colspan="2">
    <div class="tile">
      <div>
        <img src="https://wpt.fyi/static/win.svg" alt="Windows" title="Windows">
        Windows
      </div>
      <div>
        <img src="../assets/images/implstatus/chromeos.svg" alt="ChromeOS" title="ChromeOS">
        ChromeOS
      </div>
      <div>
        <img src="../assets/images/implstatus/android_robot.svg" alt="Android" title="Android">
        Android
      </div>
      <div>
        <img src="https://wpt.fyi/static/linux.svg" alt="Linux" title="Linux">
        Linux
      </div>
    </div>
  </th>
  <th colspan="2">
    <div class="tile">
      <div>
        <img src="https://wpt.fyi/static/win.svg" alt="Windows" title="Windows">
        Windows
      </div>
    </div>
  </th>
  <th colspan="2">
    <div class="tile">
      <div>
      <img src="https://wpt.fyi/static/mac.svg" alt="macOS" title="macOS">
      macOS
      </div>
    </div>
  </th>
</tr>
<tr class="title">
  <th>Operations
  </th>
  <th>
    <img src="https://raw.githubusercontent.com/alrra/browser-logos/main/src/chrome-canary/chrome-canary_128x128.png" />
    <img
      src="https://raw.githubusercontent.com/alrra/browser-logos/main/src/edge-canary/edge-canary_128x128.png" /> <sup><a href="#note-4">4</a></sup>
  </th>
  <th>Operations</th>
  <th>
    <img src="https://raw.githubusercontent.com/alrra/browser-logos/main/src/chrome-canary/chrome-canary_128x128.png" />
    <img
      src="https://raw.githubusercontent.com/alrra/browser-logos/main/src/edge-canary/edge-canary_128x128.png" /> <sup><a href="#note-5">5</a></sup>
  </th>
   <th>Operations</th>
  <th>
    <img src="https://raw.githubusercontent.com/alrra/browser-logos/main/src/chrome-canary/chrome-canary_128x128.png" /> <sup><a href="#note-6">6</a></sup>
  </th>
</tr>
LiteRT
DirectML
Core ML
✅ Supported ()
⏳ Partly Implemented ()
🚀 Work in Progress ()
❌ Not Supported
✅ Supported ()
⏳ Partly Implemented ()
🚀 Work in Progress ()
❌ Not Supported
✅ Supported ()
⏳ Partly Implemented ()
🚀 Work in Progress ()
❌ Not Supported

JavaScript ML Frameworks Integration Status

<style type="text/css"> .st0 { fill: #005A9C; }
          .st1 {
            fill: #221B0A;
          }
        </style>
        <g>
          <path class="st0" d="M118.4,76.6l24.2,82.4l24.2-82.4h17.5l-40.1,135.3h-1.7l-25.1-83.8l-25.1,83.8h-1.7L50.8,76.6h17.5L92.5,159
                l16.4-55.5l-8-26.9L118.4,76.6L118.4,76.6z" />
          <path class="st0" d="M234.2,168.5c0,12.3-3.3,22.6-9.8,30.9c-6.5,8.3-15,12.5-25.3,12.5c-7.8,0-14.6-2.5-20.4-7.4
                c-5.8-5-10.1-11.7-12.9-20.1l13.7-5.7c2,5.1,4.7,9.2,7.9,12.1c3.3,2.9,7.1,4.4,11.6,4.4c4.7,0,8.6-2.6,11.9-7.9
                c3.2-5.2,4.8-11.5,4.8-18.9c0-8.1-1.7-14.4-5.2-18.9c-4-5.2-10.3-7.9-18.9-7.9h-6.7v-8l23.4-40.4h-28.2l-7.9,13.4h-5V76.6h65.1v8.2
                l-24.7,42.6c8.7,2.8,15.3,7.9,19.7,15.2C232,149.9,234.2,158.6,234.2,168.5L234.2,168.5z" />
          <path class="st1"
            d="M302.2,75.9l2.8,17.3l-10.1,19.2c0,0-3.9-8.2-10.3-12.7c-5.4-3.8-8.9-4.6-14.4-3.5
                c-7.1,1.5-15.1,9.9-18.6,20.3c-4.2,12.5-4.2,18.5-4.4,24c-0.2,8.9,1.2,14.1,1.2,14.1s-6.1-11.3-6-27.8c0-11.8,1.9-22.5,7.4-33.1
                c4.8-9.3,11.9-14.9,18.3-15.5c6.6-0.7,11.7,2.5,15.7,5.9c4.2,3.6,8.5,11.4,8.5,11.4L302.2,75.9L302.2,75.9z" />
          <path class="st1" d="M303.4,173.6c0,0-4.4,7.9-7.2,11c-2.8,3.1-7.7,8.5-13.8,11.2c-6.1,2.7-9.3,3.2-15.4,2.6
                c-6-0.6-11.7-4.1-13.6-5.5c-2-1.4-7-5.8-9.8-9.7c-2.9-4-7.3-12-7.3-12s2.5,8,4,11.4c0.9,2,3.6,8,7.5,13.2
                c3.6,4.9,10.7,13.3,21.4,15.1c10.7,1.9,18.1-2.9,19.9-4.1c1.8-1.2,5.7-4.4,8.1-7c2.5-2.7,4.9-6.2,6.3-8.2c1-1.5,2.6-4.6,2.6-4.6
                L303.4,173.6L303.4,173.6z" />
        </g>
      </svg>
      <br />WebNN Spec
    </a>
  </th>
  <th colspan="4">JavaScript ML Frameworks Integration</th>
</tr>
<tr class="title">
  <th colspan="2">
    <a title="WebNN EP of ONNX Runtime Web" href="https://onnxruntime.ai/docs/tutorials/web/ep-webnn.html"><img src="https://onnxruntime.ai/images/svg/ONNX-Runtime-logo.svg" /></a><br />Execution Provider
    <sup><a href="#note-7">7</a></sup>
  </th>
  <th colspan="2">
      <a title="LiteRT" href="https://developers.googleblog.com/en/tensorflow-lite-is-now-litert/"><img
      src="https://storage.googleapis.com/gweb-developer-goog-blog-assets/images/LiteRT_BlogGraphics_4209x1253px_1.original.jpg" alt="LiteRT"></a>
    <br />External Delegate
    <sup><a href="#note-8">8</a></sup>
  </th>
</tr>
<tr class="title">
  <th>Operations
  </th>
  <th class="tfop">EP Version
  </th>
  <th class="tfop">Operations
  </th>
  <th class="tfop">Delegate Version
  </th>
</tr>
ONNX Runtime Web
Execution Provider
LiteRT
External Delegate
✅ Supported ()
⏳ Partly Implemented ()
🚀 Work in Progress ()
❌ Not Supported
✅ Supported ()
⏳ Partly Implemented ()
🚀 Work in Progress ()
❌ Not Supported
The total number of WebNN ops is 78. These tables currently lists ops that are implemented or work in progress by multiple backends and JavaScript machine learning frameworks.

[1] LiteRT Builtin Options
[2] DirectML API
[3] Core ML operators
[4] This feature is experimental. Can be enabled by setting #web-machine-learning-neural-network flag to Enabled.
[5] This feature is experimental. Can be enabled by setting #web-machine-learning-neural-network flag to Enabled. Supported on GPUs on Windows 11 21H2 or higher.
[6] This feature is experimental. Can be enabled by setting #web-machine-learning-neural-network flag to Enabled.
[7] ONNX Operator Schemas and WebNN EP Helper
[8] LiteRT built-in operators kTfLiteBuiltin*

🏅🏅🏅 Interested in contributing to this implementation status page? See contributing guidelines.

<script> const qS = (selector) => { return document.querySelector(selector); } const qSA = (selector) => { return document.querySelectorAll(selector); } const init = () => { let ops = document.querySelector('#ops'); let opsFramework = document.querySelector('#ops_framework'); let jsonPath = "../assets/json/webnn_status.json"; if(location.hostname.toLowerCase().indexOf('webmachinelearning.github.io') >-1) { jsonPath = "https://webmachinelearning.github.io/assets/json/webnn_status.json"; } fetch(jsonPath) .then(response => response.json()) .then(data => { let impl_status = data.impl_status; let oplist = ''; let oplistFramework = ''; for (let s of impl_status) { let spec = `${s.op}` let wpt = ''; if (s.wpt_progress === 4) { wpt = ` 📈 🧪` } if (s.wpt_progress === 3) { wpt = `🚀🚀` } let tflite = ''; if (s.tflite_op.toString().trim() === "") { tflite = `${s.tflite_chromium_version_added}`; } else if (s.tflite_progress === 4) { let tflite_ops = ''; if (s.tflite_op.length > 1) { for (let o of s.tflite_op) { tflite_ops = `✅ ${o}
`; tflite += tflite_ops; } } else if (s.tflite_op.length === 1) { tflite = '✅ ' + s.tflite_op; } else { tflite = `` } tflite = `${tflite}${s.tflite_chromium_version_added}`; } else if (s.tflite_progress === 3) { let tflite_ops = ''; if (s.tflite_op.length > 1) { for (let o of s.tflite_op) { tflite_ops = `🚀 ${o}
`; tflite += tflite_ops; } } else if (s.tflite_op.length === 1) { tflite = '🚀 ' + s.tflite_op; } else { tflite = `` } tflite = `${tflite}${s.tflite_chromium_version_added}`; } else if (s.tflite_progress === 2) { let tflite_ops = ''; if (s.tflite_op.length > 1) { for (let o of s.tflite_op) { tflite_ops = `📅 ${o}
`; tflite += tflite_ops; } } else if (s.tflite_op.length === 1) { tflite = '📅 ' + s.tflite_op; } else { tflite = `` } tflite = `${tflite}${s.tflite_chromium_version_added}`; } else { tflite = `${s.tflite_chromium_version_added}`; } let dml = ''; if (s.dml_op.toString().trim() === "") { dml = `${s.dml_chromium_version_added}`; } else if (s.dml_progress === 4) { let dml_ops = ''; if (s.dml_op.length > 1) { for (let o of s.dml_op) { dml_ops = `✅ ${o}
`; dml += dml_ops; } } else if (s.dml_op.length === 1) { dml = '✅ ' + s.dml_op; } else { dml = `` } dml = `${dml}${s.dml_chromium_version_added}`; } else if (s.dml_progress === 3) { let dml_ops = ''; if (s.dml_op.length > 1) { for (let o of s.dml_op) { dml_ops = `🚀 ${o}
`; dml += dml_ops; } } else if (s.dml_op.length === 1) { dml = '🚀 ' + s.dml_op; } else { dml = `` } dml = `${dml}${s.dml_chromium_version_added}`; } else if (s.dml_progress === 2) { let dml_ops = ''; if (s.dml_op.length > 1) { for (let o of s.dml_op) { dml_ops = `📅 ${o}
`; dml += dml_ops; } } else if (s.dml_op.length === 1) { dml = '📅 ' + s.dml_op; } else { dml = `` } dml = `${dml}${s.dml_chromium_version_added}`; } else { dml = `${s.dml_chromium_version_added}`; } let coreml = ''; if (s.coreml_op.toString().trim() === "") { coreml = `${s.coreml_chromium_version_added}`; } else if (s.coreml_progress === 4) { let coreml_ops = ''; if (s.coreml_op.length > 1) { for (let o of s.coreml_op) { coreml_ops = `✅ ${o}
`; coreml += coreml_ops; } } else if (s.coreml_op.length === 1) { coreml = '✅ ' + s.coreml_op; } else { coreml = `` } coreml = `${coreml}${s.coreml_chromium_version_added}`; } else if (s.coreml_progress === 3) { let coreml_ops = ''; if (s.coreml_op.length > 1) { for (let o of s.coreml_op) { coreml_ops = `🚀 ${o}
`; coreml += coreml_ops; } } else if (s.coreml_op.length === 1) { coreml = '🚀 ' + s.coreml_op; } else { coreml = `` } coreml = `${coreml}${s.coreml_chromium_version_added}`; } else if (s.coreml_progress === 2) { let coreml_ops = ''; if (s.coreml_op.length > 1) { for (let o of s.coreml_op) { coreml_ops = `📅 ${o}
`; coreml += coreml_ops; } } else if (s.coreml_op.length === 1) { coreml = '📅 ' + s.coreml_op; } else { coreml = `` } coreml = `${coreml}${s.coreml_chromium_version_added}`; } else { coreml = `${s.coreml_chromium_version_added}`; } let fw_tflite = ''; if (s.fw_tflite_op.toString().trim() === "") { fw_tflite = `${s.fw_tflite_version_added}`; } else if (s.fw_tflite_progress === 4) { let fw_tflite_ops = ''; if (s.fw_tflite_op.length > 1) { for (let o of s.fw_tflite_op) { fw_tflite_ops = `✅ ${o}
`; fw_tflite += fw_tflite_ops; } } else if (s.fw_tflite_op.length === 1) { fw_tflite = '✅ ' + s.fw_tflite_op; } else { fw_tflite = `` } fw_tflite = `${fw_tflite}${s.fw_tflite_version_added}`; } else if (s.fw_tflite_progress === 3) { let fw_tflite_ops = ''; if (s.fw_tflite_op.length > 1) { for (let o of s.fw_tflite_op) { fw_tflite_ops = `🚀 ${o}
`; fw_tflite += fw_tflite_ops; } } else if (s.fw_tflite_op.length === 1) { fw_tflite = '🚀 ' + s.fw_tflite_op; } else { fw_tflite = `` } fw_tflite = `${fw_tflite}${s.fw_tflite_version_added}`; } else if (s.fw_tflite_progress === 2) { let fw_tflite_ops = ''; if (s.fw_tflite_op.length > 1) { for (let o of s.fw_tflite_op) { fw_tflite_ops = `📅 ${o}
`; fw_tflite += fw_tflite_ops; } } else if (s.fw_tflite_op.length === 1) { fw_tflite = '📅 ' + s.fw_tflite_op; } else { fw_tflite = `` } fw_tflite = `${fw_tflite}${s.fw_tflite_version_added}`; } else { fw_tflite = `${s.fw_tflite_version_added}`; } let fw_ort = ''; if (s.fw_ort_op.toString().trim() === "") { fw_ort = `${s.fw_ort_version_added}`; } else if (s.fw_ort_progress === 4) { let fw_ort_ops = ''; if (s.fw_ort_op.length > 1) { for (let o of s.fw_ort_op) { fw_ort_ops = `✅ ${o}
`; fw_ort += fw_ort_ops; } } else if (s.fw_ort_op.length === 1) { fw_ort = '✅ ' + s.fw_ort_op; } else { fw_ort = `` } fw_ort = `${fw_ort}${s.fw_ort_version_added}`; } else if (s.fw_ort_progress === 3) { let fw_ort_ops = ''; if (s.fw_ort_op.length > 1) { for (let o of s.fw_ort_op) { fw_ort_ops = `🚀 ${o}
`; fw_ort += fw_ort_ops; } } else if (s.fw_ort_op.length === 1) { fw_ort = '🚀 ' + s.fw_ort_op; } else { fw_ort = `` } fw_ort = `${fw_ort}${s.fw_ort_version_added}`; } else if (s.fw_ort_progress === 2) { let fw_ort_ops = ''; if (s.fw_ort_op.length > 1) { for (let o of s.fw_ort_op) { fw_ort_ops = `📅 ${o}
`; fw_ort += fw_ort_ops; } } else if (s.fw_ort_op.length === 1) { fw_ort = '📅 ' + s.fw_ort_op; } else { fw_ort = `` } fw_ort = `${fw_ort}${s.fw_ort_version_added}`; } else { fw_ort = `${s.fw_ort_version_added}`; } oplist += ` ${spec} ${wpt} ${tflite} ${dml} ${coreml} `; oplistFramework += ` ${spec} ${fw_ort} ${fw_tflite} `; } oplist = oplist + ` `; oplistFramework = oplistFramework + ` `; ops.innerHTML = oplist; opsFramework.innerHTML = oplistFramework; count(); }) .catch(console.error); } const count = () => { let spec_defined_total = 91; let spec_s = qSA('.spec').length / 2; qS('#spec_total').innerHTML = `${spec_s}`; qS('#spec2_total').innerHTML = `${spec_s}`; let wpt_s = qSA('.wpt_s').length; let wpt_percentage = (wpt_s / spec_defined_total * 100).toFixed(1) ; qS('#wpt_total').innerHTML = `${wpt_s} / ${spec_defined_total}, ${wpt_percentage}%`; let tflite_s = qSA('.tflite_s').length; let tflite_pi = qSA('.tflite_pi').length; let tflite_wip = qSA('.tflite_wip').length; let tflite_percentage = (tflite_s / spec_defined_total * 100).toFixed(1) ; qS('#tflite_total').innerHTML = `${tflite_s} / ${spec_defined_total}, ${tflite_percentage}%`; qS('#tflite_supported').innerHTML = tflite_s; qS('#tflite_partlyimplemented').innerHTML = tflite_pi; qS('#tflite_workinprogress').innerHTML = tflite_wip; let dml_s = qSA('.dml_s').length; let dml_pi = qSA('.dml_pi').length; let dml_wip = qSA('.dml_wip').length; let dml_percentage = (dml_s / spec_defined_total * 100).toFixed(1) ; qS('#dml_total').innerHTML = `${dml_s} / ${spec_defined_total}, ${dml_percentage}%`; qS('#dml_supported').innerHTML = dml_s; qS('#dml_partlyimplemented').innerHTML = dml_pi; qS('#dml_workinprogress').innerHTML = dml_wip; let coreml_s = qSA('.coreml_s').length; let coreml_pi = qSA('.coreml_pi').length; let coreml_wip = qSA('.coreml_wip').length; let coreml_percentage = (coreml_s / spec_defined_total * 100).toFixed(1) ; qS('#coreml_total').innerHTML = `${coreml_s} / ${spec_defined_total}, ${coreml_percentage}%`; qS('#coreml_supported').innerHTML = coreml_s; qS('#coreml_partlyimplemented').innerHTML = coreml_pi; qS('#coreml_workinprogress').innerHTML = coreml_wip; let ed_s = qSA('.ed_s').length; let ed_pi = qSA('.ed_pi').length; let ed_wip = qSA('.ed_wip').length; let ed_percentage = (ed_s / spec_defined_total * 100).toFixed(1) ; qS('#ed_total').innerHTML = `${ed_s} / ${spec_defined_total}, ${ed_percentage}%`; qS('#ed_supported').innerHTML = ed_s; qS('#ed_partlyimplemented').innerHTML = ed_pi; qS('#ed_workinprogress').innerHTML = ed_wip; let ep_s = qSA('.ep_s').length; let ep_pi = qSA('.ep_pi').length; let ep_wip = qSA('.ep_wip').length; let ep_percentage = (ep_s / spec_defined_total * 100).toFixed(1) ; qS('#ep_total').innerHTML = `${ep_s} / ${spec_defined_total}, ${ep_percentage}%`; qS('#ep_supported').innerHTML = ep_s; qS('#ep_partlyimplemented').innerHTML = ep_pi; qS('#ep_workinprogress').innerHTML = ep_wip; } document.addEventListener('DOMContentLoaded', init, false); </script>