fuchsia_driver_test/
lib.rs1use anyhow::{Context as _, Result};
6use fuchsia_component_test::{Capability, ChildOptions, RealmBuilder, RealmInstance, Ref, Route};
7use {
8 fidl_fuchsia_component_test as ftest, fidl_fuchsia_driver_test as fdt, fidl_fuchsia_io as fio,
9};
10
11pub const COMPONENT_NAME: &str = "driver_test_realm";
12pub const DRIVER_TEST_REALM_URL: &str = "#meta/driver_test_realm.cm";
13
14#[async_trait::async_trait]
15pub trait DriverTestRealmBuilder {
16 async fn driver_test_realm_manifest_setup(&self, manifest_url: &str) -> Result<&Self>;
21 async fn driver_test_realm_setup(&self) -> Result<&Self>;
24
25 async fn driver_test_realm_add_dtr_exposes(
32 &self,
33 dtr_exposes: &Vec<ftest::Capability>,
34 ) -> Result<&Self>;
35
36 async fn driver_test_realm_add_dtr_offers(
43 &self,
44 dtr_offers: &Vec<ftest::Capability>,
45 from: Ref,
46 ) -> Result<&Self>;
47}
48
49#[async_trait::async_trait]
50impl DriverTestRealmBuilder for RealmBuilder {
51 async fn driver_test_realm_manifest_setup(&self, manifest_url: &str) -> Result<&Self> {
52 let driver_realm =
53 self.add_child(COMPONENT_NAME, manifest_url, ChildOptions::new().eager()).await?;
54
55 self.add_route(
59 Route::new()
60 .capability(Capability::protocol_by_name("fuchsia.logger.LogSink"))
61 .capability(Capability::protocol_by_name("fuchsia.inspect.InspectSink"))
62 .capability(Capability::protocol_by_name("fuchsia.diagnostics.ArchiveAccessor"))
63 .capability(Capability::protocol_by_name(
64 "fuchsia.component.resolution.Resolver-hermetic",
65 ))
66 .capability(Capability::protocol_by_name("fuchsia.pkg.PackageResolver-hermetic"))
67 .capability(Capability::dictionary("diagnostics"))
68 .from(Ref::parent())
69 .to(&driver_realm),
70 )
71 .await?;
72 self.add_route(
74 Route::new()
75 .capability(Capability::directory("dev-class").rights(fio::R_STAR_DIR))
76 .capability(Capability::directory("dev-topological").rights(fio::R_STAR_DIR))
77 .capability(Capability::protocol_by_name("fuchsia.system.state.Administrator"))
78 .capability(Capability::protocol_by_name("fuchsia.driver.development.Manager"))
79 .capability(Capability::protocol_by_name(
80 "fuchsia.driver.framework.CompositeNodeManager",
81 ))
82 .capability(Capability::protocol_by_name(
83 "fuchsia.driver.registrar.DriverRegistrar",
84 ))
85 .capability(Capability::protocol_by_name("fuchsia.driver.test.Realm"))
86 .from(&driver_realm)
87 .to(Ref::parent()),
88 )
89 .await?;
90 Ok(&self)
92 }
93
94 async fn driver_test_realm_setup(&self) -> Result<&Self> {
95 self.driver_test_realm_manifest_setup(DRIVER_TEST_REALM_URL).await
96 }
97
98 async fn driver_test_realm_add_dtr_exposes(
99 &self,
100 dtr_exposes: &Vec<ftest::Capability>,
101 ) -> Result<&Self> {
102 let mut decl = self.get_component_decl(COMPONENT_NAME).await?;
103 for expose in dtr_exposes {
104 let name = match expose {
105 fidl_fuchsia_component_test::Capability::Protocol(p) => p.name.as_ref(),
106 fidl_fuchsia_component_test::Capability::Directory(d) => d.name.as_ref(),
107 fidl_fuchsia_component_test::Capability::Storage(s) => s.name.as_ref(),
108 fidl_fuchsia_component_test::Capability::Service(s) => s.name.as_ref(),
109 fidl_fuchsia_component_test::Capability::EventStream(e) => e.name.as_ref(),
110 fidl_fuchsia_component_test::Capability::Config(c) => c.name.as_ref(),
111 fidl_fuchsia_component_test::Capability::Dictionary(d) => d.name.as_ref(),
112 _ => None,
113 };
114 let expose_parsed = name
115 .expect("No name found in capability.")
116 .parse::<cm_types::Name>()
117 .expect("Not a valid capability name");
118
119 decl.exposes.push(cm_rust::ExposeDecl::Service(cm_rust::ExposeServiceDecl {
120 source: cm_rust::ExposeSource::Collection(
121 "realm_builder".parse::<cm_types::Name>().unwrap(),
122 ),
123 source_name: expose_parsed.clone(),
124 source_dictionary: Default::default(),
125 target_name: expose_parsed.clone(),
126 target: cm_rust::ExposeTarget::Parent,
127 availability: cm_rust::Availability::Required,
128 }));
129 }
130 self.replace_component_decl(COMPONENT_NAME, decl).await?;
131
132 for expose in dtr_exposes {
133 self.add_route(
135 Route::new()
136 .capability(expose.clone())
137 .from(Ref::child(COMPONENT_NAME))
138 .to(Ref::parent()),
139 )
140 .await?;
141 }
142
143 Ok(&self)
144 }
145
146 async fn driver_test_realm_add_dtr_offers(
147 &self,
148 dtr_offers: &Vec<ftest::Capability>,
149 from: Ref,
150 ) -> Result<&Self> {
151 let mut decl = self.get_component_decl(COMPONENT_NAME).await?;
152 for offer in dtr_offers {
153 let name = match offer {
154 fidl_fuchsia_component_test::Capability::Protocol(p) => p.name.as_ref(),
155 fidl_fuchsia_component_test::Capability::Directory(d) => d.name.as_ref(),
156 fidl_fuchsia_component_test::Capability::Storage(s) => s.name.as_ref(),
157 fidl_fuchsia_component_test::Capability::Service(s) => s.name.as_ref(),
158 fidl_fuchsia_component_test::Capability::EventStream(e) => e.name.as_ref(),
159 fidl_fuchsia_component_test::Capability::Config(c) => c.name.as_ref(),
160 fidl_fuchsia_component_test::Capability::Dictionary(d) => d.name.as_ref(),
161 _ => None,
162 };
163 let offer_parsed = name
164 .expect("No name found in capability.")
165 .parse::<cm_types::Name>()
166 .expect("Not a valid capability name");
167
168 decl.offers.push(cm_rust::OfferDecl::Protocol(cm_rust::OfferProtocolDecl {
169 source: cm_rust::OfferSource::Parent,
170 source_name: offer_parsed.clone(),
171 source_dictionary: Default::default(),
172 target_name: offer_parsed.clone(),
173 target: cm_rust::OfferTarget::Collection(
174 "realm_builder".parse::<cm_types::Name>().unwrap(),
175 ),
176 dependency_type: cm_rust::DependencyType::Strong,
177 availability: cm_rust::Availability::Required,
178 }));
179 }
180 self.replace_component_decl(COMPONENT_NAME, decl).await?;
181
182 for offer in dtr_offers {
183 self.add_route(
185 Route::new()
186 .capability(offer.clone())
187 .from(from.clone())
188 .to(Ref::child(COMPONENT_NAME)),
189 )
190 .await?;
191 }
192
193 Ok(&self)
194 }
195}
196
197#[async_trait::async_trait]
198pub trait DriverTestRealmInstance {
199 async fn driver_test_realm_start(&self, args: fdt::RealmArgs) -> Result<()>;
201
202 fn driver_test_realm_connect_to_dev(&self) -> Result<fio::DirectoryProxy>;
204}
205
206#[async_trait::async_trait]
207impl DriverTestRealmInstance for RealmInstance {
208 async fn driver_test_realm_start(&self, args: fdt::RealmArgs) -> Result<()> {
209 let config = self.root.connect_to_protocol_at_exposed_dir::<fdt::RealmMarker>()?;
210 let () = config
211 .start(args)
212 .await
213 .context("DriverTestRealm Start failed")?
214 .map_err(zx_status::Status::from_raw)
215 .context("DriverTestRealm Start failed")?;
216 Ok(())
217 }
218
219 fn driver_test_realm_connect_to_dev(&self) -> Result<fio::DirectoryProxy> {
220 fuchsia_fs::directory::open_directory_async(
221 self.root.get_exposed_dir(),
222 "dev-topological",
223 fio::Flags::empty(),
224 )
225 .map_err(Into::into)
226 }
227}