当前位置:首页 > 科技  > 软件

WebGPU 入门:绘制一个三角形

来源: 责编: 时间:2023-11-01 09:19:00 206观看
导读大家好,我是前端西瓜哥。今天我们来入门 WebGPU,来写一个图形版本的 Hello World,即绘制一个三角形。WebGPU 是什么?WebGPU 是一个正在开发中的潜在 Web 标准和 JavaScript API,目标是提供 “现代化的 3D 图形和计算能力”

J8Z28资讯网——每日最新资讯28at.com

大家好,我是前端西瓜哥。J8Z28资讯网——每日最新资讯28at.com

今天我们来入门 WebGPU,来写一个图形版本的 Hello World,即绘制一个三角形。J8Z28资讯网——每日最新资讯28at.com

WebGPU 是什么?

WebGPU 是一个正在开发中的潜在 Web 标准和 JavaScript API,目标是提供 “现代化的 3D 图形和计算能力”。J8Z28资讯网——每日最新资讯28at.com

简单来说,WebGPU 提供一个更现代的 Web 上的图形渲染标准。J8Z28资讯网——每日最新资讯28at.com

WebGPU 的出现就是为了取代 WebGL 的,因为后者的 API 实在有些过时,无法利用好现代 GPU 的一些高级特性,本身的 API 设计也较难使用。J8Z28资讯网——每日最新资讯28at.com

相比 WebGL,WebGPU 有更好的性能表现,API 更底层更灵活,并支持更高级的现代特性,比如计算着色器。J8Z28资讯网——每日最新资讯28at.com

毫无疑问,WebGPU 是前端图形渲染的未来,值得去学习一下。J8Z28资讯网——每日最新资讯28at.com

像是以性能著称的前端图形库 PixiJS,也开始进行支持 WebGPU 的工作,并在最近发布了预览版本,声称性能将是 WebGL 的 2.5 倍。J8Z28资讯网——每日最新资讯28at.com

J8Z28资讯网——每日最新资讯28at.com

不过目前 WebGPU 还不够成熟,仍有许多工作要做,且只有少数浏览器的最新版本直接支持或通过设置开启。J8Z28资讯网——每日最新资讯28at.com

即使之后所有浏览器都支持了,旧版本浏览器还是不支持的,离大范围使用还有相当长的一段路要走。J8Z28资讯网——每日最新资讯28at.com

J8Z28资讯网——每日最新资讯28at.com

只能说未来可期。J8Z28资讯网——每日最新资讯28at.com

但生产中,我们可以做一个回退机制:如果浏览器支持 WebGPU,我们用 WebGPU 去渲染,如果不支持就回滚到 WebGL。J8Z28资讯网——每日最新资讯28at.com

只要在底层渲染方案上封装一层渲染器 renderer,就像 PixiJS 现在做的事情一样,个人还是比较期待它在性能上的提升的。J8Z28资讯网——每日最新资讯28at.com

绘制三角形

OK,我们开始用 WebGPU 绘制一个三角形。J8Z28资讯网——每日最新资讯28at.com

确保你的浏览器支持 WebGPU,建议用 Chrome,并更新到最新版本。J8Z28资讯网——每日最新资讯28at.com

这里我们创建一个宽高各为 300 的 canvas 元素,用于绘制图形。J8Z28资讯网——每日最新资讯28at.com

<canvas width="300" height="300"></canvas>

初始化 WebGPU 相关的一些对象。J8Z28资讯网——每日最新资讯28at.com

adapter 和 device

创建一个适配器对象 adapter,适配器是一个 GPU 物理硬件设备的抽象。J8Z28资讯网——每日最新资讯28at.com

const adapter = await navigator.gpu.requestAdapter();

requestAdapter() 方法会查看系统上所有可用的 GPU 设备,并选择其中合适的适配器。该方法可以传一些参数,去按条件匹配。比如 { powerPreference: 'low-power' } 表示优先使用低能耗的 GPU。J8Z28资讯网——每日最新资讯28at.com

此外,这个方法返回的是一个 Promise,即它是 异步的,需要用 await 的方式去等待异步的结果。J8Z28资讯网——每日最新资讯28at.com

然后基于 adapter,调用 requestDevice 方法拿到设备对象 device。J8Z28资讯网——每日最新资讯28at.com

device 可以理解为 adapter 的一个会话。做个比喻的话 adapter 是一个公司,device 是一个具体干活的人。J8Z28资讯网——每日最新资讯28at.com

const device = await adapter.requestDevice();

requestDevice() 方法也可以传入配置项,去开启一些高级特性,或是指定一些硬件限制,比如最大纹理尺寸。J8Z28资讯网——每日最新资讯28at.com

配置 canvas

类似 canvas 2d 和 webgl,我们需要通过 canvas 元素拿到上下文。J8Z28资讯网——每日最新资讯28at.com

const canvas = document.querySelector('canvas');const ctx = canvas.getContext('webgpu');

接着是调用 ctx.configure() 方法配置刚刚声明的 device 对象和像素格式。J8Z28资讯网——每日最新资讯28at.com

const canvasFormat = navigator.gpu.getPreferredCanvasFormat();// 给上下文配置 device 对象和ctx.configure({  device,  format: canvasFormat,});

navigator.gpu.getPreferredCanvasFormat() 会返回当前环境合适的像素格式的字符串标识,通常是 'bgra8unorm',表示用 8 位无符号整数来表示蓝色、绿色、红色和透明度四个分量。J8Z28资讯网——每日最新资讯28at.com

设置背景色

创建命令编码器 GPUCommandEncoder 实例,它用于编码需要提交给 GPU 的命令。J8Z28资讯网——每日最新资讯28at.com

const encoder = device.createCommandEncoder();

开启一个新的渲染通道(Render Pass),这里清空颜色缓冲区时填充了一个浅蓝色背景。J8Z28资讯网——每日最新资讯28at.com

和 WebGL 一样,使用 RGBA 的格式,每个分量为 0 到 1 的范围,比如 { r: 1, g: 0, b: 0, a: 1 } 表示红色,或者你可以用数组的形式 [1, 0, 0, 1]。J8Z28资讯网——每日最新资讯28at.com

const pass = encoder.beginRenderPass({  // 颜色附件,一个用于存储渲染输出颜色数据的纹理  colorAttachments: [    {      // 要渲染到的目标      view: ctx.getCurrentTexture().createView(),      // 渲染前清空颜色缓冲区      loadOp: 'clear',      // 清除颜色为浅蓝色,不设置会默认使用黑色      clearValue: { r: 0.6, g: 0.8, b: 0.9, a: 1 },      // 渲染结果会被保留在纹理中,后序好绘制到 canvas 上      storeOp: 'store',    },  ],});

我们先不绘制三角形,看看背景的渲染效果,为此我们提前执行下面代码:J8Z28资讯网——每日最新资讯28at.com

// 这里是绘制三角形的代码,之后会实现pass.end(); // 完成指令队列的记录const commandBuffer = encoder.finish(); // 结束编码device.queue.submit([commandBuffer]); // 提交给 GPU 命令队列

J8Z28资讯网——每日最新资讯28at.com

远峰蓝。J8Z28资讯网——每日最新资讯28at.com

创建缓冲区

先说说 WebGPU 的坐标系,它和 WebGL 一样,原点在画布中心,x 轴向右,y 轴向上,取值范围都是 -1 到 1。J8Z28资讯网——每日最新资讯28at.com

J8Z28资讯网——每日最新资讯28at.com

声明顶点数据。这些顶点为组成三角形的三个坐标。J8Z28资讯网——每日最新资讯28at.com

const vertices = new Float32Array([  -0.5, -0.5,  0.5, -0.5,  0.5, 0.5,]);

然后创建顶点缓冲区:J8Z28资讯网——每日最新资讯28at.com

const vertexBuffer = device.createBuffer({  // 标识,字符串随意写,报错时会通过它定位  label: 'Triangle Vertices',  // 缓冲区大小,这里是 24 字节。6 个 4 字节(即 32 位)的浮点数  size: vertices.byteLength,  // 标识缓冲区用途(1)用于顶点着色器(2)可以从 CPU 复制数据到缓冲区  usage: GPUBufferUsage.VERTEX | GPUBufferUsage.COPY_DST,});

label 方便我们定位错误位置:J8Z28资讯网——每日最新资讯28at.com

J8Z28资讯网——每日最新资讯28at.com

接着是将顶点数据复制到缓冲区:J8Z28资讯网——每日最新资讯28at.com

device.queue.writeBuffer(vertexBuffer, /* bufferOffset */ 0, vertices);

参数 bufferOffset 表示缓冲区偏移多少字节数的位置写入数据。J8Z28资讯网——每日最新资讯28at.com

读取方式

设置缓冲区的读取方式。J8Z28资讯网——每日最新资讯28at.com

const vertexBufferLayout = {  // 每组读 8 个字节。一个坐标为两个浮点数(2 * 4字节)  arrayStride: 2 * 4,   attributes: [    {      // 指定数据格式,这样 WebGPU 才知道该如何解析,格式为 2 个 32位浮点数      format: 'float32x2',      offset: 0, // 从每组的第一个数字开始      shaderLocation: 0, // 顶点着色器中的位置    },  ],};

attributes 是一个数组,这里我们只有顶点要读,所以只有一个数组元素。如果引入了颜色值并和顶点放在一起,我们就要多声明一个数组元素,并将 offset 指定到颜色的位置。J8Z28资讯网——每日最新资讯28at.com

这个对象此时还没用到,后面设置渲染流水线时会用到。J8Z28资讯网——每日最新资讯28at.com

着色器

声明 WebGPU 的着色器,创建着色器模块(GPUShaderModule)。J8Z28资讯网——每日最新资讯28at.com

WebGPU 使用特有的 WGSL 着色器语言,顶点着色器和片元着色器可以写在一起的。J8Z28资讯网——每日最新资讯28at.com

// 创建着色器模块const vertexShaderModule = device.createShaderModule({  label: 'Vertex Shader',  code: `    @vertex    fn vertexMain(@location(0) pos: vec2f) -> @builtin(position) vec4f {      return vec4f(pos, 0, 1);    }    @fragment    fn fragmentMain() -> @location(0) vec4f {      return vec4f(1, 0, 0, 1);    }  `,});

顶点着色器函数。J8Z28资讯网——每日最新资讯28at.com

@vertex fn vertexMain(@location(0) pos: vec2f) -> @builtin(position) vec4f {  return vec4f(pos, 0, 1);}
  • @vertex:装饰器,表示顶点着色器主函数。
  • @location(0):缓冲区读取方式设置的 shaderLocation,这里拿到了两个浮点数。
  • vec2f:两个浮点数的向量,同理,vec4f 为 4 浮点数的向量。
  • -> @builtin(position):表示函数的返回值会被设置为内置的顶点位置变量。WebGPU 是利用函数的返回值配合修饰符的方式进行内部变量赋值的。

片元着色器。J8Z28资讯网——每日最新资讯28at.com

@fragmentfn fragmentMain() -> @location(0) vec4f {  return vec4f(1, 0, 0, 1); // 红色}
  • @fragment 表示片元着色器主函数。
  • -> @location(0) 表示将返回的颜色输出到位置为 0 的颜色附件上,简单来说,就是给对应点设置为对应颜色。

渲染流水线

创建渲染流水线,也就是把之前的设置组合起来,用哪个着色器的哪个函数作为入口、如何读取缓冲区等。J8Z28资讯网——每日最新资讯28at.com

const pipeline = device.createRenderPipeline({  label: 'pipeline', // 标识,定位错误用  layout: 'auto', // 自动流水线布局  vertex: {    module: vertexShaderModule, // 着色器模块    entryPoint: 'vertexMain', // 入口函数为 vertexMain    buffers: [vertexBufferLayout], // 读取缓冲区的方式  },  fragment: {    module: vertexShaderModule,    entryPoint: 'fragmentMain',    targets: [      {        format: canvasFormat, // 输出到 canvas 画布上      },    ],  },});

将渲染流水线设置到 pass 上。J8Z28资讯网——每日最新资讯28at.com

pass.setPipeline(pipeline);

将缓冲区绑定到管线的第一个顶点缓冲槽(slot)。J8Z28资讯网——每日最新资讯28at.com

pass.setVertexBuffer(0, vertexBuffer);

绘制图元,这里要设置绘制几组,一组是两个点,所以要处以 2。J8Z28资讯网——每日最新资讯28at.com

pass.draw(vertices.length / 2);

然后就是前面讲过的收尾代码。J8Z28资讯网——每日最新资讯28at.com

pass.end(); // 完成指令队列的记录const commandBuffer = encoder.finish(); // 结束编码device.queue.submit([commandBuffer]); // 提交给 GPU 命令队列

至此,一个三角形就画好了。J8Z28资讯网——每日最新资讯28at.com

绘制结果

J8Z28资讯网——每日最新资讯28at.com

完整代码

线上 demo 演示:J8Z28资讯网——每日最新资讯28at.com

https://codesandbox.io/s/lg4w27?file=/src/index.mjs。J8Z28资讯网——每日最新资讯28at.com

完整代码:J8Z28资讯网——每日最新资讯28at.com

const render = async () => {  const adapter = await navigator.gpu.requestAdapter();  const device = await adapter.requestDevice();  const canvas = document.querySelector('canvas');  const ctx = canvas.getContext('webgpu');  const canvasFormat = navigator.gpu.getPreferredCanvasFormat();  ctx.configure({    device,    format: canvasFormat,  });  const encoder = device.createCommandEncoder();  const pass = encoder.beginRenderPass({    colorAttachments: [      {        view: ctx.getCurrentTexture().createView(),        loadOp: 'clear',        clearValue: { r: 0.6, g: 0.8, b: 0.9, a: 1 },        storeOp: 'store',      },    ],  });  // 创建顶点数据  // prettier-ignore  const vertices = new Float32Array([    -0.5, -0.5,    0.5, -0.5,    0.5, 0.5,  ]);  // 缓冲区  const vertexBuffer = device.createBuffer({    // 标识,字符串随意写,报错时会通过它定位,    label: 'Triangle Vertices',    // 缓冲区大小,这里是 24 字节。6 个 4 字节(即 32 位)的浮点数    size: vertices.byteLength,    // 标识缓冲区用途(1)用于顶点着色器(2)可以从 CPU 复制数据到缓冲区    // eslint-disable-next-line no-undef    usage: GPUBufferUsage.VERTEX | GPUBufferUsage.COPY_DST,  });  // 将顶点数据复制到缓冲区  device.queue.writeBuffer(vertexBuffer, /* bufferOffset */ 0, vertices);  // GPU 应该如何读取缓冲区中的数据  const vertexBufferLayout = {    arrayStride: 2 * 4, // 每一组的字节数,每组有两个数字(2 * 4字节)    attributes: [      {        format: 'float32x2', // 每个数字是32位浮点数        offset: 0, // 从每组的第一个数字开始        shaderLocation: 0, // 顶点着色器中的位置      },    ],  };  // 着色器用的是 WGSL 着色器语言  const vertexShaderModule = device.createShaderModule({    label: 'Vertex Shader',    code: `      @vertex      fn vertexMain(@location(0) pos: vec2f) -> @builtin(position) vec4f {        return vec4f(pos, 0, 1);      }      @fragment      fn fragmentMain() -> @location(0) vec4f {        return vec4f(1, 0, 0, 1);      }    `,  });  // 渲染流水线  const pipeline = device.createRenderPipeline({    label: 'pipeline',    layout: 'auto',    vertex: {      module: vertexShaderModule,      entryPoint: 'vertexMain',      buffers: [vertexBufferLayout],    },    fragment: {      module: vertexShaderModule,      entryPoint: 'fragmentMain',      targets: [        {          format: canvasFormat,        },      ],    },  });  pass.setPipeline(pipeline);  pass.setVertexBuffer(0, vertexBuffer);  pass.draw(vertices.length / 2);  pass.end();  const commandBuffer = encoder.finish();  device.queue.submit([commandBuffer]);};render();

结尾

本文讲解了如何用 WebGPU 绘制一个三角形。可以看到它和 WebGL 的逻辑有很多共同之处的,都要创建缓冲区、着色器、定义读取方式。J8Z28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-16287-0.htmlWebGPU 入门:绘制一个三角形

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 掌握Spring事件监听器的内部逻辑与实现

下一篇: 转转Flutter实践之路

标签:
  • 热门焦点
  • 一加Ace2 Pro真机揭晓 钛空灰配色质感拉满

    一加Ace2 Pro真机揭晓 钛空灰配色质感拉满

    终于,在经过了几波预热之后,一加Ace2 Pro的外观真机图在网上出现了。还是博主数码闲聊站曝光的,这次的外观设计还是延续了一加11的方案,只是细节上有了调整,例如新加入了钛空灰
  • 影音体验是真的强 简单聊聊iQOO Pad

    影音体验是真的强 简单聊聊iQOO Pad

    大公司的好处就是产品线丰富,非常细分化的东西也能给你做出来,例如早先我们看到了新的vivo Pad2,之后我们又在iQOO Neo8 Pro的发布会上看到了iQOO的首款平板产品iQOO Pad。虽
  • Redmi Buds 4开箱简评:才199还有降噪 可以无脑入

    Redmi Buds 4开箱简评:才199还有降噪 可以无脑入

    在上个月举办的Redmi Note11T Pro系列新机发布会上,除了两款手机新品之外,Redmi还带来了两款TWS真无线蓝牙耳机产品,Redmi Buds 4和Redmi Buds 4 Pro,此前我们在Redmi Note11T
  • Automa-通过连接块来自动化你的浏览器

    Automa-通过连接块来自动化你的浏览器

    1、前言通过浏览器插件可实现自动化脚本的录制与编写,具有代表性的工具就是:Selenium IDE、Katalon Recorder,对于简单的业务来说可快速实现自动化的上手工作。Selenium IDEKat
  • Golang 中的 io 包详解:组合接口

    Golang 中的 io 包详解:组合接口

    io.ReadWriter// ReadWriter is the interface that groups the basic Read and Write methods.type ReadWriter interface { Reader Writer}是对Reader和Writer接口的组合,
  • Python异步IO编程的进程/线程通信实现

    Python异步IO编程的进程/线程通信实现

    这篇文章再讲3种方式,同时讲4中进程间通信的方式一、 Python 中线程间通信的实现方式共享变量共享变量是多个线程可以共同访问的变量。在Python中,可以使用threading模块中的L
  • 年轻人的“职场羞耻感”,无处不在

    年轻人的“职场羞耻感”,无处不在

    作者:冯晓亭 陶 淘 李 欣 张 琳 马舒叶来源:燃次元&ldquo;人在职场,应该选择什么样的着装?&rdquo;近日,在网络上,一个与着装相关的帖子引发关注,在该帖子里,一位在高级写字楼亚洲金
  • 阿里瓴羊One推出背后,零售企业迎数字化新解

    阿里瓴羊One推出背后,零售企业迎数字化新解

    作者:刘旷近年来随着数字经济的高速发展,各式各样的SaaS应用服务更是层出不穷,但本质上SaaS大多局限于单一业务流层面,对用户核心关切的增长问题等则没有提供更好的解法。在Saa
  • iQOO Neo8 Pro抢先上架:首发天玑9200+ 安卓性能之王

    iQOO Neo8 Pro抢先上架:首发天玑9200+ 安卓性能之王

    经过了一段时间的密集爆料,昨日iQOO官方如期对外宣布:将于5月23日推出全新的iQOO Neo8系列新品,官方称这是一款拥有旗舰级性能调校的作品。随着发布时
Top