blob: 4586ccad969fb751fdca3201246b8aca18edadcf [file] [log] [blame]
Bob Badoure8af2842022-04-07 10:44:22 -07001package {
2 default_applicable_licenses: ["Android-Apache-2.0"],
3}
4
Andrew Walbran68a8c162022-03-07 15:38:42 +00005rust_ffi_static {
6 name: "libpvmfw",
7 crate_name: "pvmfw",
David Brazdil30a7f082022-07-07 15:30:14 +01008 defaults: ["vmbase_ffi_defaults"],
Andrew Walbran68a8c162022-03-07 15:38:42 +00009 srcs: ["src/main.rs"],
Pierre-Clément Tosie8726e42022-10-17 13:35:27 +010010 features: [
11 "legacy",
12 ],
Andrew Walbranf2594882022-03-15 17:32:53 +000013 rustlibs: [
Pierre-Clément Tosia8a4a202022-11-03 14:16:46 +000014 "libaarch64_paging",
Alice Wang947f3f72023-09-29 09:04:07 +000015 "libbssl_avf_nostd",
Shikha Panwar8f7fc1a2024-04-10 10:41:34 +000016 "libcbor_util_nostd",
Alan Stokesd1ee63e2023-04-24 16:17:39 +010017 "libciborium_nostd",
18 "libciborium_io_nostd",
Pierre-Clément Tosi1bf532b2023-11-13 11:06:20 +000019 "libcstr",
Alice Wangcb9d2f92023-02-06 10:29:00 +000020 "libdiced_open_dice_nostd",
Pierre-Clément Tosi992c2bb2024-10-02 10:37:42 +010021 "liblibfdt_nostd",
Pierre-Clément Tosi37105a62022-10-18 12:21:48 +010022 "liblog_rust_nostd",
Pierre-Clément Tosi2d652982023-01-09 19:30:17 +000023 "libpvmfw_avb_nostd",
Pierre-Clément Tosi4ef75222022-10-26 17:40:50 +010024 "libpvmfw_embedded_key",
Jiyong Park216793e2023-02-25 02:15:44 +090025 "libpvmfw_fdt_template",
Alice Wang12e4c862023-12-11 13:52:27 +000026 "libservice_vm_version",
Andrew Walbranf44f1602023-05-30 14:59:19 +000027 "libsmccc",
Pierre-Clément Tosi1cc5eb72023-02-02 11:09:18 +000028 "libstatic_assertions",
Pierre-Clément Tosi328dfb62022-11-25 18:20:42 +000029 "libtinyvec_nostd",
Pierre-Clément Tosi1cc5eb72023-02-02 11:09:18 +000030 "libuuid_nostd",
Andrew Walbrand1d03182022-12-09 18:20:01 +000031 "libvirtio_drivers",
Andrew Walbrandd74b902022-04-14 16:12:50 +000032 "libvmbase",
Alan Stokesa0e42962023-04-14 17:59:50 +010033 "libzerocopy_nostd",
Pierre-Clément Tosi8383c542022-11-01 14:07:29 +000034 "libzeroize_nostd",
Andrew Walbranf2594882022-03-15 17:32:53 +000035 ],
Andrew Walbran68a8c162022-03-07 15:38:42 +000036}
Andrew Walbran15068b02022-03-22 15:57:34 +000037
Alice Wang1c7e0eb2023-04-11 17:03:42 +000038// Generates an empty file.
39genrule {
40 name: "empty_file",
41 out: ["empty_file"],
42 cmd: "touch $(out)",
43}
44
Alan Stokesddb988c2023-11-27 11:13:06 +000045rust_defaults {
46 name: "libpvmfw.test.defaults",
Nikita Ioffeda1b2732023-09-04 13:46:56 +010047 defaults: ["avf_build_flags_rust"],
Jiyong Parkc5d2ef22023-04-11 01:23:46 +090048 test_suites: ["general-tests"],
49 test_options: {
50 unit_test: true,
51 },
Alan Stokesddb988c2023-11-27 11:13:06 +000052 prefer_rlib: true,
Jiyong Parkc5d2ef22023-04-11 01:23:46 +090053 rustlibs: [
Pierre-Clément Tosi1bf532b2023-11-13 11:06:20 +000054 "libcstr",
Alan Stokesddb988c2023-11-27 11:13:06 +000055 ],
56}
57
58rust_test {
59 name: "libpvmfw.bootargs.test",
60 host_supported: true,
61 // For now, only bootargs.rs is written to be conditionally compiled with std.
62 srcs: ["src/bootargs.rs"],
63 defaults: ["libpvmfw.test.defaults"],
64 rustlibs: [
Jiyong Parkc5d2ef22023-04-11 01:23:46 +090065 "libzeroize",
66 ],
67}
68
Alan Stokesddb988c2023-11-27 11:13:06 +000069rust_test {
70 name: "libpvmfw.device_assignment.test",
71 srcs: ["src/device_assignment.rs"],
72 defaults: ["libpvmfw.test.defaults"],
73 rustlibs: [
Jaewan Kim8f6f4662023-12-12 17:38:47 +090074 "libdts",
Alan Stokesddb988c2023-11-27 11:13:06 +000075 "liblibfdt",
76 "liblog_rust",
77 "libpvmfw_fdt_template",
Jaewan Kim8f6f4662023-12-12 17:38:47 +090078 "libzerocopy",
Alan Stokesddb988c2023-11-27 11:13:06 +000079 ],
80 data: [
81 ":test_pvmfw_devices_vm_dtbo",
82 ":test_pvmfw_devices_vm_dtbo_without_symbols",
Jaewan Kim19b984f2023-12-04 15:16:50 +090083 ":test_pvmfw_devices_vm_dtbo_with_duplicated_iommus",
Pierre-Clément Tosi49e26ce2024-03-12 16:31:50 +000084 ":test_pvmfw_devices_overlapping_pvmfw",
Jaewan Kim8f6f4662023-12-12 17:38:47 +090085 ":test_pvmfw_devices_vm_dtbo_with_dependencies",
Alan Stokesddb988c2023-11-27 11:13:06 +000086 ":test_pvmfw_devices_with_rng",
Alan Stokesddb988c2023-11-27 11:13:06 +000087 ":test_pvmfw_devices_with_multiple_devices_iommus",
88 ":test_pvmfw_devices_with_iommu_sharing",
89 ":test_pvmfw_devices_with_iommu_id_conflict",
Jaewan Kim52477ae2023-11-21 21:20:52 +090090 ":test_pvmfw_devices_without_device",
Jaewan Kima67e36a2023-11-29 16:50:23 +090091 ":test_pvmfw_devices_without_iommus",
Jaewan Kim19b984f2023-12-04 15:16:50 +090092 ":test_pvmfw_devices_with_duplicated_pviommus",
93 ":test_pvmfw_devices_with_multiple_reg_iommus",
Jaewan Kim8f6f4662023-12-12 17:38:47 +090094 ":test_pvmfw_devices_with_dependency",
95 ":test_pvmfw_devices_with_dependency_loop",
96 ":test_pvmfw_devices_with_multiple_dependencies",
97 ":test_pvmfw_devices_expected_dt",
Alan Stokesddb988c2023-11-27 11:13:06 +000098 ],
Jaewan Kim8f6f4662023-12-12 17:38:47 +090099 data_bins: ["dtc_static"],
100 compile_multilib: "first",
Alan Stokesddb988c2023-11-27 11:13:06 +0000101 // To use libpvmfw_fdt_template for testing
102 enabled: false,
103 target: {
104 android_arm64: {
105 enabled: true,
106 },
107 },
108}
109
110rust_test {
111 name: "libpvmfw.dice.test",
112 srcs: ["src/dice.rs"],
113 defaults: ["libpvmfw.test.defaults"],
114 rustlibs: [
115 "libcbor_util",
116 "libciborium",
Shikha Panwarf3acfd12024-05-28 15:48:13 +0000117 "libdiced_open_dice",
Alan Stokesddb988c2023-11-27 11:13:06 +0000118 "libpvmfw_avb_nostd",
Alan Stokesa17cfba2024-02-14 17:34:51 +0000119 "libzerocopy_nostd",
Shikha Panwarf3acfd12024-05-28 15:48:13 +0000120 "libhex",
Alan Stokesddb988c2023-11-27 11:13:06 +0000121 ],
122}
123
Jaewan Kimc6e023b2023-10-12 15:11:05 +0900124genrule {
125 name: "test_pvmfw_devices_vm_dtbo",
126 defaults: ["dts_to_dtb"],
127 srcs: ["testdata/test_pvmfw_devices_vm_dtbo.dts"],
128 out: ["test_pvmfw_devices_vm_dtbo.dtbo"],
129}
130
131genrule {
132 name: "test_pvmfw_devices_vm_dtbo_without_symbols",
133 defaults: ["dts_to_dtb"],
134 srcs: ["testdata/test_pvmfw_devices_vm_dtbo_without_symbols.dts"],
135 out: ["test_pvmfw_devices_vm_dtbo_without_symbols.dtbo"],
136}
137
Jaewan Kim19b984f2023-12-04 15:16:50 +0900138genrule {
139 name: "test_pvmfw_devices_vm_dtbo_with_duplicated_iommus",
140 defaults: ["dts_to_dtb"],
141 srcs: ["testdata/test_pvmfw_devices_vm_dtbo_with_duplicated_iommus.dts"],
142 out: ["test_pvmfw_devices_vm_dtbo_with_duplicated_iommus.dtbo"],
143}
144
Jaewan Kim8f6f4662023-12-12 17:38:47 +0900145genrule {
146 name: "test_pvmfw_devices_vm_dtbo_with_dependencies",
147 tools: ["dtc"],
148 cmd: "$(location dtc) -@ -I dts -O dtb $(in) -o $(out)",
149 srcs: ["testdata/test_pvmfw_devices_vm_dtbo_with_dependencies.dts"],
150 out: ["test_pvmfw_devices_vm_dtbo_with_dependencies.dtbo"],
151}
152
Jaewan Kimcb0d45d2023-11-30 15:40:12 +0900153genrule_defaults {
154 name: "test_device_assignment_dts_to_dtb",
155 defaults: ["dts_to_dtb"],
156 srcs: ["testdata/test_crosvm_dt_base.dtsi"],
157}
158
Jaewan Kimc6e023b2023-10-12 15:11:05 +0900159genrule {
Pierre-Clément Tosi49e26ce2024-03-12 16:31:50 +0000160 name: "test_pvmfw_devices_overlapping_pvmfw",
161 defaults: ["test_device_assignment_dts_to_dtb"],
162 srcs: ["testdata/test_pvmfw_devices_overlapping_pvmfw.dts"],
163 out: ["test_pvmfw_devices_overlapping_pvmfw.dtb"],
164}
165
166genrule {
Jaewan Kimc6e023b2023-10-12 15:11:05 +0900167 name: "test_pvmfw_devices_with_rng",
Jaewan Kimcb0d45d2023-11-30 15:40:12 +0900168 defaults: ["test_device_assignment_dts_to_dtb"],
Jaewan Kimc6e023b2023-10-12 15:11:05 +0900169 srcs: ["testdata/test_pvmfw_devices_with_rng.dts"],
170 out: ["test_pvmfw_devices_with_rng.dtb"],
171}
172
Jaewan Kim51ccfed2023-11-08 13:51:58 +0900173genrule {
Jaewan Kima67e36a2023-11-29 16:50:23 +0900174 name: "test_pvmfw_devices_without_iommus",
Jaewan Kimcb0d45d2023-11-30 15:40:12 +0900175 defaults: ["test_device_assignment_dts_to_dtb"],
Jaewan Kima67e36a2023-11-29 16:50:23 +0900176 srcs: ["testdata/test_pvmfw_devices_without_iommus.dts"],
177 out: ["test_pvmfw_devices_without_iommus.dtb"],
Jaewan Kim51ccfed2023-11-08 13:51:58 +0900178}
179
180genrule {
Jaewan Kim52477ae2023-11-21 21:20:52 +0900181 name: "test_pvmfw_devices_without_device",
182 defaults: ["test_device_assignment_dts_to_dtb"],
183 srcs: ["testdata/test_pvmfw_devices_without_device.dts"],
184 out: ["test_pvmfw_devices_without_device.dtb"],
185}
186
187genrule {
Jaewan Kim51ccfed2023-11-08 13:51:58 +0900188 name: "test_pvmfw_devices_with_multiple_devices_iommus",
Jaewan Kimcb0d45d2023-11-30 15:40:12 +0900189 defaults: ["test_device_assignment_dts_to_dtb"],
Jaewan Kim51ccfed2023-11-08 13:51:58 +0900190 srcs: ["testdata/test_pvmfw_devices_with_multiple_devices_iommus.dts"],
191 out: ["test_pvmfw_devices_with_multiple_devices_iommus.dtb"],
192}
193
194genrule {
195 name: "test_pvmfw_devices_with_iommu_sharing",
Jaewan Kimcb0d45d2023-11-30 15:40:12 +0900196 defaults: ["test_device_assignment_dts_to_dtb"],
Jaewan Kim51ccfed2023-11-08 13:51:58 +0900197 srcs: ["testdata/test_pvmfw_devices_with_iommu_sharing.dts"],
198 out: ["test_pvmfw_devices_with_iommu_sharing.dtb"],
199}
200
201genrule {
202 name: "test_pvmfw_devices_with_iommu_id_conflict",
Jaewan Kimcb0d45d2023-11-30 15:40:12 +0900203 defaults: ["test_device_assignment_dts_to_dtb"],
Jaewan Kim51ccfed2023-11-08 13:51:58 +0900204 srcs: ["testdata/test_pvmfw_devices_with_iommu_id_conflict.dts"],
205 out: ["test_pvmfw_devices_with_iommu_id_conflict.dtb"],
206}
207
Jaewan Kim19b984f2023-12-04 15:16:50 +0900208genrule {
209 name: "test_pvmfw_devices_with_duplicated_pviommus",
210 defaults: ["test_device_assignment_dts_to_dtb"],
211 srcs: ["testdata/test_pvmfw_devices_with_duplicated_pviommus.dts"],
212 out: ["test_pvmfw_devices_with_duplicated_pviommus.dtb"],
213}
214
215genrule {
216 name: "test_pvmfw_devices_with_multiple_reg_iommus",
217 defaults: ["test_device_assignment_dts_to_dtb"],
218 srcs: ["testdata/test_pvmfw_devices_with_multiple_reg_iommus.dts"],
219 out: ["test_pvmfw_devices_with_multiple_reg_iommus.dtb"],
220}
221
Jaewan Kim8f6f4662023-12-12 17:38:47 +0900222genrule {
223 name: "test_pvmfw_devices_with_dependency",
224 defaults: ["test_device_assignment_dts_to_dtb"],
225 srcs: ["testdata/test_pvmfw_devices_with_dependency.dts"],
226 out: ["test_pvmfw_devices_with_dependency.dtb"],
227}
228
229genrule {
230 name: "test_pvmfw_devices_with_multiple_dependencies",
231 defaults: ["test_device_assignment_dts_to_dtb"],
232 srcs: ["testdata/test_pvmfw_devices_with_multiple_dependencies.dts"],
233 out: ["test_pvmfw_devices_with_multiple_dependencies.dtb"],
234}
235
236genrule {
237 name: "test_pvmfw_devices_with_dependency_loop",
238 defaults: ["test_device_assignment_dts_to_dtb"],
239 srcs: ["testdata/test_pvmfw_devices_with_dependency_loop.dts"],
240 out: ["test_pvmfw_devices_with_dependency_loop.dtb"],
241}
242
243// We can't use genrule because preprocessed platform DT is built with cc_object.
244// cc_genrule doesn't support default, so we'll build all expected DTs in
245// a single build rule.
246cc_genrule {
247 name: "test_pvmfw_devices_expected_dt",
248 srcs: [
249 ":pvmfw_platform.dts.preprocessed",
250 "testdata/expected_dt_with_dependency.dts",
251 "testdata/expected_dt_with_multiple_dependencies.dts",
252 "testdata/expected_dt_with_dependency_loop.dts",
253 ],
254 out: [
255 "expected_dt_with_dependency.dtb",
256 "expected_dt_with_multiple_dependencies.dtb",
257 "expected_dt_with_dependency_loop.dtb",
258 ],
259 tools: ["dtc"],
260 cmd: "FILES=($(in));" +
261 "cp $${FILES[0]} $(genDir)/platform_preprocessed.dts;" +
262 "for DTS in $${FILES[@]:1}; do" +
263 " DTB=$$(basename -s .dts $${DTS}).dtb;" +
264 " $(location dtc) -@ -i $(genDir) -I dts -O dtb $${DTS} -o $(genDir)/$${DTB};" +
265 "done",
266 visibility: ["//visibility:private"],
267}
268
Andrew Walbran15068b02022-03-22 15:57:34 +0000269cc_binary {
Pierre-Clément Tosib1300352022-09-09 11:01:06 +0100270 name: "pvmfw",
David Brazdil30a7f082022-07-07 15:30:14 +0100271 defaults: ["vmbase_elf_defaults"],
Andrew Walbran15068b02022-03-22 15:57:34 +0000272 srcs: [
Andrew Walbran267f6c12022-03-24 11:26:36 +0000273 "idmap.S",
Andrew Walbran15068b02022-03-22 15:57:34 +0000274 ],
275 static_libs: [
276 "libpvmfw",
277 ],
Andrew Walbrana5b7af52022-07-06 15:06:20 +0000278 linker_scripts: [
279 "image.ld",
280 ":vmbase_sections",
281 ],
Jiyong Park17b8d752022-12-16 14:36:24 +0900282 // `installable: false` is inherited from vmbase_elf_defaults, and that
283 // hides this module from Make, which makes it impossible for the Make world
284 // to place the unstripped binary to the symbols directory. Marking back as
285 // installable exposes this module to the Make world again. Note that this
286 // module (pvmfw) still is NOT installed to any of the filesystem images. It
287 // is fed into pvmfw_bin and then into pvmfw_img to become a standalone
288 // partition image. This is just to package the unstripped file into the
289 // symbols zip file for debugging purpose.
290 installable: true,
Andrew Walbran15068b02022-03-22 15:57:34 +0000291}
David Brazdil05d4e072022-04-25 14:47:06 +0100292
293raw_binary {
Pierre-Clément Tosib1300352022-09-09 11:01:06 +0100294 name: "pvmfw_bin",
295 stem: "pvmfw.bin",
296 src: ":pvmfw",
David Brazdil05d4e072022-04-25 14:47:06 +0100297 enabled: false,
298 target: {
299 android_arm64: {
300 enabled: true,
301 },
302 },
303}
David Brazdilac216b52022-04-25 15:07:22 +0100304
Jaewan Kim2cf6f392023-02-10 01:35:47 +0900305// Provide pvmfw.bin binary regardless of the architecture for building test.
306// Note that skipping tests on unsupported device is easy
307// while configuring server configuration to make such tests to run on working
308// devices.
309prebuilt_etc {
310 name: "pvmfw_test",
311 filename: "pvmfw_test.bin",
312 target: {
313 android_arm64: {
314 src: ":pvmfw_bin",
315 },
316 },
Alice Wang1c7e0eb2023-04-11 17:03:42 +0000317 src: ":empty_file",
Jaewan Kim2cf6f392023-02-10 01:35:47 +0900318 installable: false,
319}
320
Pierre-Clément Tosi8b5533d2024-05-24 10:47:59 +0100321filegroup {
Pierre-Clément Tosi6ec2ae22022-10-26 15:14:45 +0100322 name: "pvmfw_embedded_key",
Pierre-Clément Tosi8b5533d2024-05-24 10:47:59 +0100323 srcs: [":avb_testkey_rsa4096"],
324}
325
326genrule {
327 name: "pvmfw_embedded_key_pub_bin",
328 tools: ["avbtool"],
329 srcs: [":pvmfw_embedded_key"],
330 out: ["pvmfw_embedded_key_pub.bin"],
331 cmd: "$(location avbtool) extract_public_key --key $(in) --output $(out)",
Pierre-Clément Tosi6ec2ae22022-10-26 15:14:45 +0100332}
333
Pierre-Clément Tosi4ef75222022-10-26 17:40:50 +0100334genrule {
335 name: "pvmfw_embedded_key_rs",
Pierre-Clément Tosi8b5533d2024-05-24 10:47:59 +0100336 srcs: [":pvmfw_embedded_key_pub_bin"],
Pierre-Clément Tosi4ef75222022-10-26 17:40:50 +0100337 out: ["lib.rs"],
338 cmd: "(" +
339 " echo '#![no_std]';" +
340 " echo '#![allow(missing_docs)]';" +
341 " echo 'pub const PUBLIC_KEY: &[u8] = &[';" +
342 " xxd -i < $(in);" +
343 " echo '];';" +
344 ") > $(out)",
345}
346
347rust_library_rlib {
348 name: "libpvmfw_embedded_key",
Pierre-Clément Tosi4894b432023-06-27 14:02:26 +0000349 defaults: ["vmbase_rlib_defaults"],
Pierre-Clément Tosi4ef75222022-10-26 17:40:50 +0100350 srcs: [":pvmfw_embedded_key_rs"],
351 crate_name: "pvmfw_embedded_key",
Pierre-Clément Tosi4ef75222022-10-26 17:40:50 +0100352}
353
Pierre-Clément Tosi6ec2ae22022-10-26 15:14:45 +0100354prebuilt_etc {
Pierre-Clément Tosib5771c02022-09-06 16:01:35 +0100355 name: "pvmfw_sign_key",
356 src: ":avb_testkey_rsa4096",
357 installable: false,
358}
359
Jiyong Park9c1c8182023-05-25 15:40:11 +0900360// We need to rename *.dts into *.cpp as cc_object doesn't accept *.dts as an
361// input
Jiyong Park216793e2023-02-25 02:15:44 +0900362genrule {
Jiyong Park9c1c8182023-05-25 15:40:11 +0900363 name: "pvmfw_platform.dts.renamed",
364 srcs: ["platform.dts"],
365 out: ["out.cpp"],
366 cmd: "cp $(in) $(out)",
367 visibility: ["//visibility:private"],
368}
369
370// Then run the macro processor to replace symbols like GIC_SPI into actual
371// numbers defined in the ARM DT binding headers
372cc_object {
373 name: "pvmfw_platform.dts.preprocessed",
Nikita Ioffe38b9e712024-02-08 15:55:07 +0000374 defaults: ["avf_build_flags_cc"],
Jiyong Park9c1c8182023-05-25 15:40:11 +0900375 header_libs: ["arm_dt_bindings_headers"],
376 host_supported: true,
377 srcs: [":pvmfw_platform.dts.renamed"],
378 cflags: [
379 "-E",
380 "-P",
381 "-xassembler-with-cpp", // allow C preprocessor directives
Jiyong Park216793e2023-02-25 02:15:44 +0900382 ],
Jiyong Park9c1c8182023-05-25 15:40:11 +0900383 visibility: ["//visibility:private"],
384}
385
386// Compile the preprocessed dts into binary and create a rust library source
387// having the binary.
388cc_genrule {
389 name: "pvmfw_fdt_template_rs",
390 srcs: [":pvmfw_platform.dts.preprocessed"],
Jiyong Park216793e2023-02-25 02:15:44 +0900391 out: ["lib.rs"],
392 tools: ["dtc"],
Jaewan Kim96411e92023-11-13 15:54:02 +0900393 cmd: "$(location dtc) -@ -I dts -O dtb -o $(genDir)/compiled.dtbo $(in) && " +
Jiyong Park216793e2023-02-25 02:15:44 +0900394 "(" +
395 " echo '#![no_std]';" +
396 " echo '#![allow(missing_docs)]';" +
397 " echo 'pub const RAW: &[u8] = &[';" +
398 " xxd -i < $(genDir)/compiled.dtbo;" +
399 " echo '];';" +
400 ") > $(out)",
Jiyong Park9c1c8182023-05-25 15:40:11 +0900401 visibility: ["//visibility:private"],
Jiyong Park216793e2023-02-25 02:15:44 +0900402}
403
404rust_library_rlib {
405 name: "libpvmfw_fdt_template",
Pierre-Clément Tosi4894b432023-06-27 14:02:26 +0000406 defaults: ["vmbase_rlib_defaults"],
Jiyong Park216793e2023-02-25 02:15:44 +0900407 srcs: [":pvmfw_fdt_template_rs"],
408 crate_name: "pvmfw_fdt_template",
409}
410
David Brazdilac216b52022-04-25 15:07:22 +0100411bootimg {
412 name: "pvmfw_img",
413 stem: "pvmfw.img",
414 kernel_prebuilt: ":pvmfw_bin",
415 header_version: "3",
416 partition_name: "pvmfw",
417 enabled: false,
418 target: {
419 android_arm64: {
420 enabled: true,
421 },
422 },
Pierre-Clément Tosib5771c02022-09-06 16:01:35 +0100423 use_avb: true,
424 avb_private_key: ":pvmfw_sign_key",
David Brazdilac216b52022-04-25 15:07:22 +0100425}