1use std::rc::Rc;
6
7use dom_struct::dom_struct;
8use js::context::JSContext;
9use js::jsapi::HandleObject;
10use js::realm::CurrentRealm;
11use script_bindings::reflector::{Reflector, reflect_dom_object_with_cx};
12use servo_constellation_traits::ScriptToConstellationMessage;
13use webgpu_traits::WebGPUAdapterResponse;
14use wgpu_types::PowerPreference;
15
16use super::wgsllanguagefeatures::WGSLLanguageFeatures;
17use crate::dom::bindings::codegen::Bindings::WebGPUBinding::{
18 GPUMethods, GPUPowerPreference, GPURequestAdapterOptions, GPUTextureFormat,
19};
20use crate::dom::bindings::error::Error;
21use crate::dom::bindings::reflector::DomGlobal;
22use crate::dom::bindings::root::{DomRoot, MutNullableDom};
23use crate::dom::bindings::str::DOMString;
24use crate::dom::globalscope::GlobalScope;
25use crate::dom::promise::Promise;
26use crate::dom::webgpu::gpuadapter::GPUAdapter;
27use crate::routed_promise::{RoutedPromiseListener, callback_promise};
28
29#[dom_struct]
30#[expect(clippy::upper_case_acronyms)]
31pub(crate) struct GPU {
32 reflector_: Reflector,
33 wgsl_language_features: MutNullableDom<WGSLLanguageFeatures>,
35}
36
37impl GPU {
38 pub(crate) fn new_inherited() -> GPU {
39 GPU {
40 reflector_: Reflector::new(),
41 wgsl_language_features: MutNullableDom::default(),
42 }
43 }
44
45 pub(crate) fn new(cx: &mut JSContext, global: &GlobalScope) -> DomRoot<GPU> {
46 reflect_dom_object_with_cx(Box::new(GPU::new_inherited()), global, cx)
47 }
48}
49
50impl GPUMethods<crate::DomTypeHolder> for GPU {
51 fn RequestAdapter(
53 &self,
54 cx: &mut CurrentRealm,
55 options: &GPURequestAdapterOptions,
56 ) -> Rc<Promise> {
57 let global = &self.global();
58 let promise = Promise::new_in_realm(cx);
60 let task_manager = global.task_manager();
61 let task_source = task_manager.dom_manipulation_task_source();
62 let callback = callback_promise(&promise, self, task_source);
63
64 let power_preference = match options.powerPreference {
65 Some(GPUPowerPreference::Low_power) => PowerPreference::LowPower,
66 Some(GPUPowerPreference::High_performance) => PowerPreference::HighPerformance,
67 None => PowerPreference::default(),
68 };
69 let ids = global.wgpu_id_hub().create_adapter_id();
70
71 match &*options.featureLevel.str() {
82 "core" => {},
83 "compatibility" => {
84 },
87 _ => {
88 promise.resolve_native(cx, &None::<GPUAdapter>);
89 return promise;
90 },
91 }
92 let script_to_constellation_chan = global.script_to_constellation_chan();
93 if script_to_constellation_chan
94 .send(ScriptToConstellationMessage::RequestAdapter(
95 callback,
96 wgpu_core::instance::RequestAdapterOptions {
97 power_preference,
98 compatible_surface: None,
99 force_fallback_adapter: options.forceFallbackAdapter,
100 },
101 ids,
102 ))
103 .is_err()
104 {
105 promise.reject_error(cx, Error::Operation(None));
106 }
107 promise
109 }
110
111 fn GetPreferredCanvasFormat(&self) -> GPUTextureFormat {
113 if cfg!(target_os = "android") {
115 GPUTextureFormat::Rgba8unorm
116 } else {
117 GPUTextureFormat::Bgra8unorm
118 }
119 }
120
121 fn WgslLanguageFeatures(
123 &self,
124 cx: &mut js::context::JSContext,
125 ) -> DomRoot<WGSLLanguageFeatures> {
126 self.wgsl_language_features
127 .or_init(|| WGSLLanguageFeatures::new(cx, &self.global(), None))
128 }
129}
130
131impl RoutedPromiseListener<WebGPUAdapterResponse> for GPU {
132 fn handle_response(
133 &self,
134 cx: &mut js::context::JSContext,
135 response: WebGPUAdapterResponse,
136 promise: &Rc<Promise>,
137 ) {
138 match response {
139 Some(Ok(adapter)) => {
140 let adapter = GPUAdapter::new(
141 cx,
142 &self.global(),
143 adapter.channel,
144 DOMString::from(format!(
145 "{} ({:?})",
146 adapter.adapter_info.name, adapter.adapter_id.0
147 )),
148 HandleObject::null(),
149 adapter.features,
150 adapter.limits,
151 adapter.adapter_info,
152 adapter.adapter_id,
153 );
154 promise.resolve_native(cx, &adapter);
155 },
156 Some(Err(e)) => {
157 warn!("Could not get GPUAdapter ({:?})", e);
158 promise.resolve_native(cx, &None::<GPUAdapter>);
159 },
160 None => {
161 warn!("Couldn't get a response, because WebGPU is disabled");
162 promise.resolve_native(cx, &None::<GPUAdapter>);
163 },
164 }
165 }
166}