blob: d4bc6de401a1cd2bcab032154a7305def97b1c56 [file] [log] [blame]
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +01001/*
Biswarup Pal6152a302023-12-19 12:44:09 +00002 * Copyright 2023 The Android Open Source Project
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +01003 *
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 <memory>
18
19#include "VirtualCameraProvider.h"
20#include "aidl/android/hardware/camera/common/CameraDeviceStatus.h"
21#include "aidl/android/hardware/camera/common/Status.h"
22#include "aidl/android/hardware/camera/common/TorchModeStatus.h"
23#include "aidl/android/hardware/camera/provider/BnCameraProviderCallback.h"
24#include "android/binder_auto_utils.h"
25#include "android/binder_interface_utils.h"
26#include "gmock/gmock.h"
27#include "gtest/gtest.h"
28#include "util/Util.h"
29
30namespace android {
31namespace companion {
32namespace virtualcamera {
33namespace {
34
Jan Sebechlebsky3b478c42023-11-23 13:15:56 +010035using ::aidl::android::companion::virtualcamera::Format;
Biswarup Pal112458f2023-12-28 19:50:17 +000036using ::aidl::android::companion::virtualcamera::LensFacing;
Biswarup Pal6152a302023-12-19 12:44:09 +000037using ::aidl::android::companion::virtualcamera::SensorOrientation;
Jan Sebechlebsky3b478c42023-11-23 13:15:56 +010038using ::aidl::android::companion::virtualcamera::SupportedStreamConfiguration;
Biswarup Pal6152a302023-12-19 12:44:09 +000039using ::aidl::android::companion::virtualcamera::VirtualCameraConfiguration;
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +010040using ::aidl::android::hardware::camera::common::CameraDeviceStatus;
41using ::aidl::android::hardware::camera::common::Status;
42using ::aidl::android::hardware::camera::common::TorchModeStatus;
43using ::aidl::android::hardware::camera::provider::BnCameraProviderCallback;
44using ::testing::_;
45using ::testing::ElementsAre;
46using ::testing::Eq;
47using ::testing::IsEmpty;
48using ::testing::IsNull;
49using ::testing::MatchesRegex;
50using ::testing::Not;
51using ::testing::Return;
52
Biswarup Pal15dcd132024-06-02 14:02:03 +000053constexpr char kCameraId[] = "42";
Jan Sebechlebsky3b478c42023-11-23 13:15:56 +010054constexpr int kVgaWidth = 640;
55constexpr int kVgaHeight = 480;
Biswarup Pal6152a302023-12-19 12:44:09 +000056constexpr int kMaxFps = 30;
Biswarup Pal37a75182024-01-16 15:53:35 +000057constexpr int kDefaultDeviceId = 0;
Biswarup Pal15dcd132024-06-02 14:02:03 +000058constexpr char kVirtualCameraNameRegex[] = "device@[0-9]+\\.[0-9]+/virtual/.+";
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +010059
60class MockCameraProviderCallback : public BnCameraProviderCallback {
61 public:
62 MOCK_METHOD(ndk::ScopedAStatus, cameraDeviceStatusChange,
63 (const std::string&, CameraDeviceStatus), (override));
64 MOCK_METHOD(ndk::ScopedAStatus, torchModeStatusChange,
65 (const std::string&, TorchModeStatus), (override));
66 MOCK_METHOD(ndk::ScopedAStatus, physicalCameraDeviceStatusChange,
67 (const std::string&, const std::string&, CameraDeviceStatus),
68 (override));
69};
70
71class VirtualCameraProviderTest : public ::testing::Test {
72 public:
73 void SetUp() override {
74 mCameraProvider = ndk::SharedRefBase::make<VirtualCameraProvider>();
75 mMockCameraProviderCallback =
76 ndk::SharedRefBase::make<MockCameraProviderCallback>();
77 ON_CALL(*mMockCameraProviderCallback, cameraDeviceStatusChange)
78 .WillByDefault([](const std::string&, CameraDeviceStatus) {
79 return ndk::ScopedAStatus::ok();
80 });
81 }
82
83 protected:
84 std::shared_ptr<VirtualCameraProvider> mCameraProvider;
85 std::shared_ptr<MockCameraProviderCallback> mMockCameraProviderCallback =
86 ndk::SharedRefBase::make<MockCameraProviderCallback>();
Biswarup Pal6152a302023-12-19 12:44:09 +000087 VirtualCameraConfiguration mInputConfig = VirtualCameraConfiguration{
88 .supportedStreamConfigs = {SupportedStreamConfiguration{
89 .width = kVgaWidth,
90 .height = kVgaHeight,
91 .pixelFormat = Format::YUV_420_888,
92 .maxFps = kMaxFps}},
93 .virtualCameraCallback = nullptr,
Biswarup Pal112458f2023-12-28 19:50:17 +000094 .sensorOrientation = SensorOrientation::ORIENTATION_0,
95 .lensFacing = LensFacing::FRONT};
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +010096};
97
98TEST_F(VirtualCameraProviderTest, SetNullCameraCallbackFails) {
99 // Attempting to set callback to nullptr should fail.
100 EXPECT_FALSE(mCameraProvider->setCallback(nullptr).isOk());
101}
102
103TEST_F(VirtualCameraProviderTest, NoCamerasInitially) {
104 std::vector<std::string> cameras;
105
106 // Initially, the camera provider should return empty list
107 // of cameras.
108 ASSERT_TRUE(mCameraProvider->getCameraIdList(&cameras).isOk());
109 EXPECT_THAT(cameras, IsEmpty());
110}
111
112TEST_F(VirtualCameraProviderTest, CreateCamera) {
113 // When new camera is created, we expect
114 // cameraDeviceStatusChange to be called exactly once with
115 // PRESENT status.
116 EXPECT_CALL(*mMockCameraProviderCallback,
117 cameraDeviceStatusChange(_, CameraDeviceStatus::PRESENT))
118 .WillOnce(Return(ndk::ScopedAStatus::ok()));
119
120 ASSERT_TRUE(mCameraProvider->setCallback(mMockCameraProviderCallback).isOk());
Jan Sebechlebsky3b478c42023-11-23 13:15:56 +0100121 std::shared_ptr<VirtualCameraDevice> camera =
Biswarup Pal37a75182024-01-16 15:53:35 +0000122 mCameraProvider->createCamera(mInputConfig, kCameraId, kDefaultDeviceId);
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +0100123 EXPECT_THAT(camera, Not(IsNull()));
124 EXPECT_THAT(camera->getCameraName(), MatchesRegex(kVirtualCameraNameRegex));
125
126 // Created camera should be in the list of cameras.
127 std::vector<std::string> cameraIds;
128 ASSERT_TRUE(mCameraProvider->getCameraIdList(&cameraIds).isOk());
129 EXPECT_THAT(cameraIds, ElementsAre(camera->getCameraName()));
130}
131
132TEST_F(VirtualCameraProviderTest, CreateCameraBeforeCallbackIsSet) {
133 // We expect cameraDeviceStatusChange to be invoked even when the
134 // setCallback configures the callback after camera is already created.
135 EXPECT_CALL(*mMockCameraProviderCallback,
136 cameraDeviceStatusChange(_, CameraDeviceStatus::PRESENT))
137 .WillOnce(Return(ndk::ScopedAStatus::ok()));
138
Jan Sebechlebsky3b478c42023-11-23 13:15:56 +0100139 std::shared_ptr<VirtualCameraDevice> camera =
Biswarup Pal37a75182024-01-16 15:53:35 +0000140 mCameraProvider->createCamera(mInputConfig, kCameraId, kDefaultDeviceId);
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +0100141 ASSERT_TRUE(mCameraProvider->setCallback(mMockCameraProviderCallback).isOk());
142
143 // Created camera should be in the list of cameras.
144 std::vector<std::string> cameraIds;
145 EXPECT_TRUE(mCameraProvider->getCameraIdList(&cameraIds).isOk());
146 EXPECT_THAT(cameraIds, ElementsAre(camera->getCameraName()));
147}
148
149TEST_F(VirtualCameraProviderTest, RemoveCamera) {
150 ASSERT_TRUE(mCameraProvider->setCallback(mMockCameraProviderCallback).isOk());
Jan Sebechlebsky3b478c42023-11-23 13:15:56 +0100151 std::shared_ptr<VirtualCameraDevice> camera =
Biswarup Pal37a75182024-01-16 15:53:35 +0000152 mCameraProvider->createCamera(mInputConfig, kCameraId, kDefaultDeviceId);
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +0100153
154 EXPECT_CALL(*mMockCameraProviderCallback,
155 cameraDeviceStatusChange(Eq(camera->getCameraName()),
156 CameraDeviceStatus::NOT_PRESENT))
157 .WillOnce(Return(ndk::ScopedAStatus::ok()));
158 EXPECT_TRUE(mCameraProvider->removeCamera(camera->getCameraName()));
159
160 // There are no cameras present after only camera is removed.
161 std::vector<std::string> cameraIds;
162 ASSERT_TRUE(mCameraProvider->getCameraIdList(&cameraIds).isOk());
163 EXPECT_THAT(cameraIds, IsEmpty());
164}
165
166TEST_F(VirtualCameraProviderTest, RemoveNonExistingCamera) {
167 ASSERT_TRUE(mCameraProvider->setCallback(mMockCameraProviderCallback).isOk());
Jan Sebechlebsky3b478c42023-11-23 13:15:56 +0100168 std::shared_ptr<VirtualCameraDevice> camera =
Biswarup Pal37a75182024-01-16 15:53:35 +0000169 mCameraProvider->createCamera(mInputConfig, kCameraId, kDefaultDeviceId);
Jan Sebechlebsky5cb39962023-11-22 17:33:07 +0100170
171 // Removing non-existing camera should fail.
172 const std::string cameraName = "DefinitelyNoTCamera";
173 EXPECT_FALSE(mCameraProvider->removeCamera(cameraName));
174
175 // Camera should be still present in the camera list.
176 std::vector<std::string> cameraIds;
177 ASSERT_TRUE(mCameraProvider->getCameraIdList(&cameraIds).isOk());
178 EXPECT_THAT(cameraIds, ElementsAre(camera->getCameraName()));
179}
180
181} // namespace
182} // namespace virtualcamera
183} // namespace companion
184} // namespace android