1 /*
2 * Copyright 2020 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include <android-base/file.h>
18 #include <android/hardware/sensors/2.1/types.h>
19 #include <gtest/gtest.h>
20 #include <sys/stat.h>
21 #include <fstream>
22 #include <iostream>
23 #include <map>
24 #include <sstream>
25
26 #include "iio_utils.h"
27
28 using ::android::hardware::sensors::V2_1::SensorType;
29 using android::hardware::sensors::V2_1::subhal::implementation::iio_device_data;
30 using android::hardware::sensors::V2_1::subhal::implementation::load_iio_devices;
31 using android::hardware::sensors::V2_1::subhal::implementation::sensors_supported_hal;
32
sensorFilter(iio_device_data * dev)33 static bool sensorFilter(iio_device_data* dev) {
34 static std::map<std::string, SensorType> KNOWN_SENSORS = {
35 {"scmi.iio.accel", SensorType::ACCELEROMETER},
36 };
37
38 if (!dev) return false;
39
40 const auto iter = KNOWN_SENSORS.find(dev->name);
41 if (iter == KNOWN_SENSORS.end()) return false;
42
43 dev->type = iter->second;
44 return true;
45 }
46
TEST(IIoUtilsTest,ScanEmptyDirectory)47 TEST(IIoUtilsTest, ScanEmptyDirectory) {
48 TemporaryDir td;
49 std::vector<iio_device_data> iio_devices;
50 const auto err = load_iio_devices(td.path, &iio_devices, sensorFilter);
51 ASSERT_EQ(0, err);
52 ASSERT_EQ(0, iio_devices.size());
53 }
54
concatPaths(const std::string & a,const std::string & b)55 static std::string concatPaths(const std::string& a, const std::string& b) {
56 std::stringstream ss;
57 ss << a << '/' << b;
58 return ss.str();
59 }
60
61 template <typename T>
writeFile(const std::string & path,const T & content,bool nl=true)62 static bool writeFile(const std::string& path, const T& content, bool nl = true) {
63 std::stringstream ss;
64 ss << content;
65
66 std::ofstream f;
67 f.open(path);
68 if (!f) return false;
69 f << ss.str();
70 if (nl) f << '\n';
71 f.close();
72 return true;
73 }
74
75 template <typename U>
writeFile(const std::string & path,const std::vector<U> & content,bool nl=true)76 static bool writeFile(const std::string& path, const std::vector<U>& content, bool nl = true) {
77 std::stringstream ss;
78 bool first = true;
79 for (const auto& item : content) {
80 if (!first) ss << ' ';
81 ss << item;
82 if (first) first = false;
83 }
84
85 return writeFile(path, ss.str(), nl);
86 }
87
writeAccelDevice(const std::string & td_path,const iio_device_data & dev)88 static bool writeAccelDevice(const std::string& td_path, const iio_device_data& dev) {
89 std::stringstream ss;
90 ss << concatPaths(td_path, "iio:device") << std::to_string(dev.iio_dev_num);
91 const std::string dev_path(ss.str());
92
93 int err = mkdir(dev_path.c_str(), 0777);
94 if (err != 0) return false;
95
96 if (!writeFile(concatPaths(dev_path, "name"), dev.name)) return false;
97 if (!writeFile(concatPaths(dev_path, "in_accel_x_scale"), dev.scale)) return false;
98 if (!writeFile(concatPaths(dev_path, "in_accel_y_scale"), dev.scale)) return false;
99 if (!writeFile(concatPaths(dev_path, "in_accel_z_scale"), dev.scale)) return false;
100 if (!writeFile(concatPaths(dev_path, "in_accel_raw_available"),
101 "[-78381056.000000000 2392.000000000 78378664.000000000]"))
102 return false;
103 if (!writeFile(concatPaths(dev_path, "in_accel_sampling_frequency_available"),
104 dev.sampling_freq_avl))
105 return false;
106
107 return true;
108 }
109
110 // sets up a new iio:device<id> device with default parameters for an accelerometer
createDefaultAccelerometerDevice(int id)111 static iio_device_data createDefaultAccelerometerDevice(int id) {
112 iio_device_data dev;
113 dev.type = SensorType::ACCELEROMETER;
114 dev.iio_dev_num = id;
115 dev.name = "scmi.iio.accel";
116 dev.sampling_freq_avl = {12.500000, 26.000364, 52.002080, 104.004160, 208.003993};
117 dev.resolution = 2392;
118 dev.scale = 0.000001000f;
119 dev.max_range = 78378664;
120
121 return dev;
122 }
123
TEST(IioUtilsTest,LoadValidSensor)124 TEST(IioUtilsTest, LoadValidSensor) {
125 TemporaryDir td;
126 const std::string td_path(td.path);
127 const auto dev_model = createDefaultAccelerometerDevice(0);
128 bool ok = writeAccelDevice(td_path, dev_model);
129 ASSERT_TRUE(ok);
130
131 std::vector<iio_device_data> iio_devices;
132 const auto err = load_iio_devices(td_path, &iio_devices, sensorFilter);
133 ASSERT_EQ(0, err);
134 ASSERT_EQ(1, iio_devices.size());
135
136 const auto& accel(iio_devices[0]);
137
138 EXPECT_EQ(SensorType::ACCELEROMETER, accel.type);
139 EXPECT_EQ("scmi.iio.accel", accel.name);
140 EXPECT_EQ(0, accel.iio_dev_num);
141
142 EXPECT_NEAR(dev_model.resolution, accel.resolution, 0.0002);
143 EXPECT_NEAR(dev_model.scale, accel.scale, 0.0002);
144 EXPECT_EQ(dev_model.max_range, accel.max_range);
145
146 EXPECT_EQ(dev_model.sampling_freq_avl.size(), accel.sampling_freq_avl.size());
147 for (size_t i = 0; i < dev_model.sampling_freq_avl.size(); ++i) {
148 if (i >= accel.sampling_freq_avl.size()) break;
149 EXPECT_NEAR(dev_model.sampling_freq_avl[i], accel.sampling_freq_avl[i], 0.0002);
150 }
151 }
152