1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker * Copyright (C) 2016 The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker *
4*4d7e907cSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker *
8*4d7e907cSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker *
10*4d7e907cSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker */
16*4d7e907cSAndroid Build Coastguard Worker
17*4d7e907cSAndroid Build Coastguard Worker #define LOG_TAG "[email protected]"
18*4d7e907cSAndroid Build Coastguard Worker #include <android-base/logging.h>
19*4d7e907cSAndroid Build Coastguard Worker
20*4d7e907cSAndroid Build Coastguard Worker #include "TvInput.h"
21*4d7e907cSAndroid Build Coastguard Worker
22*4d7e907cSAndroid Build Coastguard Worker namespace android {
23*4d7e907cSAndroid Build Coastguard Worker namespace hardware {
24*4d7e907cSAndroid Build Coastguard Worker namespace tv {
25*4d7e907cSAndroid Build Coastguard Worker namespace input {
26*4d7e907cSAndroid Build Coastguard Worker namespace V1_0 {
27*4d7e907cSAndroid Build Coastguard Worker namespace implementation {
28*4d7e907cSAndroid Build Coastguard Worker
29*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_OTHER_HARDWARE == static_cast<int>(TvInputType::OTHER),
30*4d7e907cSAndroid Build Coastguard Worker "TvInputType::OTHER must match legacy value.");
31*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_TUNER == static_cast<int>(TvInputType::TUNER),
32*4d7e907cSAndroid Build Coastguard Worker "TvInputType::TUNER must match legacy value.");
33*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_COMPOSITE == static_cast<int>(TvInputType::COMPOSITE),
34*4d7e907cSAndroid Build Coastguard Worker "TvInputType::COMPOSITE must match legacy value.");
35*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_SVIDEO == static_cast<int>(TvInputType::SVIDEO),
36*4d7e907cSAndroid Build Coastguard Worker "TvInputType::SVIDEO must match legacy value.");
37*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_SCART == static_cast<int>(TvInputType::SCART),
38*4d7e907cSAndroid Build Coastguard Worker "TvInputType::SCART must match legacy value.");
39*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_COMPONENT == static_cast<int>(TvInputType::COMPONENT),
40*4d7e907cSAndroid Build Coastguard Worker "TvInputType::COMPONENT must match legacy value.");
41*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_VGA == static_cast<int>(TvInputType::VGA),
42*4d7e907cSAndroid Build Coastguard Worker "TvInputType::VGA must match legacy value.");
43*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_DVI == static_cast<int>(TvInputType::DVI),
44*4d7e907cSAndroid Build Coastguard Worker "TvInputType::DVI must match legacy value.");
45*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_HDMI == static_cast<int>(TvInputType::HDMI),
46*4d7e907cSAndroid Build Coastguard Worker "TvInputType::HDMI must match legacy value.");
47*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_TYPE_DISPLAY_PORT == static_cast<int>(TvInputType::DISPLAY_PORT),
48*4d7e907cSAndroid Build Coastguard Worker "TvInputType::DISPLAY_PORT must match legacy value.");
49*4d7e907cSAndroid Build Coastguard Worker
50*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_EVENT_DEVICE_AVAILABLE == static_cast<int>(
51*4d7e907cSAndroid Build Coastguard Worker TvInputEventType::DEVICE_AVAILABLE),
52*4d7e907cSAndroid Build Coastguard Worker "TvInputEventType::DEVICE_AVAILABLE must match legacy value.");
53*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_EVENT_DEVICE_UNAVAILABLE == static_cast<int>(
54*4d7e907cSAndroid Build Coastguard Worker TvInputEventType::DEVICE_UNAVAILABLE),
55*4d7e907cSAndroid Build Coastguard Worker "TvInputEventType::DEVICE_UNAVAILABLE must match legacy value.");
56*4d7e907cSAndroid Build Coastguard Worker static_assert(TV_INPUT_EVENT_STREAM_CONFIGURATIONS_CHANGED == static_cast<int>(
57*4d7e907cSAndroid Build Coastguard Worker TvInputEventType::STREAM_CONFIGURATIONS_CHANGED),
58*4d7e907cSAndroid Build Coastguard Worker "TvInputEventType::STREAM_CONFIGURATIONS_CHANGED must match legacy value.");
59*4d7e907cSAndroid Build Coastguard Worker
60*4d7e907cSAndroid Build Coastguard Worker sp<ITvInputCallback> TvInput::mCallback = nullptr;
61*4d7e907cSAndroid Build Coastguard Worker
TvInput(tv_input_device_t * device)62*4d7e907cSAndroid Build Coastguard Worker TvInput::TvInput(tv_input_device_t* device) : mDevice(device) {
63*4d7e907cSAndroid Build Coastguard Worker mCallbackOps.notify = &TvInput::notify;
64*4d7e907cSAndroid Build Coastguard Worker }
65*4d7e907cSAndroid Build Coastguard Worker
~TvInput()66*4d7e907cSAndroid Build Coastguard Worker TvInput::~TvInput() {
67*4d7e907cSAndroid Build Coastguard Worker if (mDevice != nullptr) {
68*4d7e907cSAndroid Build Coastguard Worker free(mDevice);
69*4d7e907cSAndroid Build Coastguard Worker }
70*4d7e907cSAndroid Build Coastguard Worker }
71*4d7e907cSAndroid Build Coastguard Worker
72*4d7e907cSAndroid Build Coastguard Worker // Methods from ::android::hardware::tv_input::V1_0::ITvInput follow.
setCallback(const sp<ITvInputCallback> & callback)73*4d7e907cSAndroid Build Coastguard Worker Return<void> TvInput::setCallback(const sp<ITvInputCallback>& callback) {
74*4d7e907cSAndroid Build Coastguard Worker mCallback = callback;
75*4d7e907cSAndroid Build Coastguard Worker if (mCallback != nullptr) {
76*4d7e907cSAndroid Build Coastguard Worker mDevice->initialize(mDevice, &mCallbackOps, nullptr);
77*4d7e907cSAndroid Build Coastguard Worker }
78*4d7e907cSAndroid Build Coastguard Worker return Void();
79*4d7e907cSAndroid Build Coastguard Worker }
80*4d7e907cSAndroid Build Coastguard Worker
getStreamConfigurations(int32_t deviceId,getStreamConfigurations_cb cb)81*4d7e907cSAndroid Build Coastguard Worker Return<void> TvInput::getStreamConfigurations(int32_t deviceId, getStreamConfigurations_cb cb) {
82*4d7e907cSAndroid Build Coastguard Worker int32_t configCount = 0;
83*4d7e907cSAndroid Build Coastguard Worker const tv_stream_config_t* configs = nullptr;
84*4d7e907cSAndroid Build Coastguard Worker int ret = mDevice->get_stream_configurations(mDevice, deviceId, &configCount, &configs);
85*4d7e907cSAndroid Build Coastguard Worker Result res = Result::UNKNOWN;
86*4d7e907cSAndroid Build Coastguard Worker hidl_vec<TvStreamConfig> tvStreamConfigs;
87*4d7e907cSAndroid Build Coastguard Worker if (ret == 0) {
88*4d7e907cSAndroid Build Coastguard Worker res = Result::OK;
89*4d7e907cSAndroid Build Coastguard Worker tvStreamConfigs.resize(getSupportedConfigCount(configCount, configs));
90*4d7e907cSAndroid Build Coastguard Worker int32_t pos = 0;
91*4d7e907cSAndroid Build Coastguard Worker for (int32_t i = 0; i < configCount; ++i) {
92*4d7e907cSAndroid Build Coastguard Worker if (isSupportedStreamType(configs[i].type)) {
93*4d7e907cSAndroid Build Coastguard Worker tvStreamConfigs[pos].streamId = configs[i].stream_id;
94*4d7e907cSAndroid Build Coastguard Worker tvStreamConfigs[pos].maxVideoWidth = configs[i].max_video_width;
95*4d7e907cSAndroid Build Coastguard Worker tvStreamConfigs[pos].maxVideoHeight = configs[i].max_video_height;
96*4d7e907cSAndroid Build Coastguard Worker ++pos;
97*4d7e907cSAndroid Build Coastguard Worker }
98*4d7e907cSAndroid Build Coastguard Worker }
99*4d7e907cSAndroid Build Coastguard Worker } else if (ret == -EINVAL) {
100*4d7e907cSAndroid Build Coastguard Worker res = Result::INVALID_ARGUMENTS;
101*4d7e907cSAndroid Build Coastguard Worker }
102*4d7e907cSAndroid Build Coastguard Worker cb(res, tvStreamConfigs);
103*4d7e907cSAndroid Build Coastguard Worker return Void();
104*4d7e907cSAndroid Build Coastguard Worker }
105*4d7e907cSAndroid Build Coastguard Worker
openStream(int32_t deviceId,int32_t streamId,openStream_cb cb)106*4d7e907cSAndroid Build Coastguard Worker Return<void> TvInput::openStream(int32_t deviceId, int32_t streamId, openStream_cb cb) {
107*4d7e907cSAndroid Build Coastguard Worker tv_stream_t stream;
108*4d7e907cSAndroid Build Coastguard Worker stream.stream_id = streamId;
109*4d7e907cSAndroid Build Coastguard Worker int ret = mDevice->open_stream(mDevice, deviceId, &stream);
110*4d7e907cSAndroid Build Coastguard Worker Result res = Result::UNKNOWN;
111*4d7e907cSAndroid Build Coastguard Worker native_handle_t* sidebandStream = nullptr;
112*4d7e907cSAndroid Build Coastguard Worker if (ret == 0) {
113*4d7e907cSAndroid Build Coastguard Worker if (isSupportedStreamType(stream.type)) {
114*4d7e907cSAndroid Build Coastguard Worker res = Result::OK;
115*4d7e907cSAndroid Build Coastguard Worker sidebandStream = stream.sideband_stream_source_handle;
116*4d7e907cSAndroid Build Coastguard Worker }
117*4d7e907cSAndroid Build Coastguard Worker } else {
118*4d7e907cSAndroid Build Coastguard Worker if (ret == -EBUSY) {
119*4d7e907cSAndroid Build Coastguard Worker res = Result::NO_RESOURCE;
120*4d7e907cSAndroid Build Coastguard Worker } else if (ret == -EEXIST) {
121*4d7e907cSAndroid Build Coastguard Worker res = Result::INVALID_STATE;
122*4d7e907cSAndroid Build Coastguard Worker } else if (ret == -EINVAL) {
123*4d7e907cSAndroid Build Coastguard Worker res = Result::INVALID_ARGUMENTS;
124*4d7e907cSAndroid Build Coastguard Worker }
125*4d7e907cSAndroid Build Coastguard Worker }
126*4d7e907cSAndroid Build Coastguard Worker cb(res, sidebandStream);
127*4d7e907cSAndroid Build Coastguard Worker return Void();
128*4d7e907cSAndroid Build Coastguard Worker }
129*4d7e907cSAndroid Build Coastguard Worker
closeStream(int32_t deviceId,int32_t streamId)130*4d7e907cSAndroid Build Coastguard Worker Return<Result> TvInput::closeStream(int32_t deviceId, int32_t streamId) {
131*4d7e907cSAndroid Build Coastguard Worker int ret = mDevice->close_stream(mDevice, deviceId, streamId);
132*4d7e907cSAndroid Build Coastguard Worker Result res = Result::UNKNOWN;
133*4d7e907cSAndroid Build Coastguard Worker if (ret == 0) {
134*4d7e907cSAndroid Build Coastguard Worker res = Result::OK;
135*4d7e907cSAndroid Build Coastguard Worker } else if (ret == -ENOENT) {
136*4d7e907cSAndroid Build Coastguard Worker res = Result::INVALID_STATE;
137*4d7e907cSAndroid Build Coastguard Worker } else if (ret == -EINVAL) {
138*4d7e907cSAndroid Build Coastguard Worker res = Result::INVALID_ARGUMENTS;
139*4d7e907cSAndroid Build Coastguard Worker }
140*4d7e907cSAndroid Build Coastguard Worker return res;
141*4d7e907cSAndroid Build Coastguard Worker }
142*4d7e907cSAndroid Build Coastguard Worker
143*4d7e907cSAndroid Build Coastguard Worker // static
notify(struct tv_input_device * __unused,tv_input_event_t * event,void * optionalStatus)144*4d7e907cSAndroid Build Coastguard Worker void TvInput::notify(struct tv_input_device* __unused, tv_input_event_t* event,
145*4d7e907cSAndroid Build Coastguard Worker void* optionalStatus) {
146*4d7e907cSAndroid Build Coastguard Worker if (mCallback != nullptr && event != nullptr) {
147*4d7e907cSAndroid Build Coastguard Worker // Capturing is no longer supported.
148*4d7e907cSAndroid Build Coastguard Worker if (event->type >= TV_INPUT_EVENT_CAPTURE_SUCCEEDED) {
149*4d7e907cSAndroid Build Coastguard Worker return;
150*4d7e907cSAndroid Build Coastguard Worker }
151*4d7e907cSAndroid Build Coastguard Worker TvInputEvent tvInputEvent;
152*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.type = static_cast<TvInputEventType>(event->type);
153*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.deviceInfo.deviceId = event->device_info.device_id;
154*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.deviceInfo.type = static_cast<TvInputType>(
155*4d7e907cSAndroid Build Coastguard Worker event->device_info.type);
156*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.deviceInfo.portId = event->device_info.hdmi.port_id;
157*4d7e907cSAndroid Build Coastguard Worker CableConnectionStatus connectionStatus = CableConnectionStatus::UNKNOWN;
158*4d7e907cSAndroid Build Coastguard Worker if (optionalStatus != nullptr &&
159*4d7e907cSAndroid Build Coastguard Worker ((event->type == TV_INPUT_EVENT_STREAM_CONFIGURATIONS_CHANGED) ||
160*4d7e907cSAndroid Build Coastguard Worker (event->type == TV_INPUT_EVENT_DEVICE_AVAILABLE))) {
161*4d7e907cSAndroid Build Coastguard Worker int newStatus = *reinterpret_cast<int*>(optionalStatus);
162*4d7e907cSAndroid Build Coastguard Worker if (newStatus <= static_cast<int>(CableConnectionStatus::DISCONNECTED) &&
163*4d7e907cSAndroid Build Coastguard Worker newStatus >= static_cast<int>(CableConnectionStatus::UNKNOWN)) {
164*4d7e907cSAndroid Build Coastguard Worker connectionStatus = static_cast<CableConnectionStatus>(newStatus);
165*4d7e907cSAndroid Build Coastguard Worker }
166*4d7e907cSAndroid Build Coastguard Worker }
167*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.deviceInfo.cableConnectionStatus = connectionStatus;
168*4d7e907cSAndroid Build Coastguard Worker // TODO: Ensure the legacy audio type code is the same once audio HAL default
169*4d7e907cSAndroid Build Coastguard Worker // implementation is ready.
170*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.deviceInfo.audioType = static_cast<AudioDevice>(
171*4d7e907cSAndroid Build Coastguard Worker event->device_info.audio_type);
172*4d7e907cSAndroid Build Coastguard Worker memset(tvInputEvent.deviceInfo.audioAddress.data(), 0,
173*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.deviceInfo.audioAddress.size());
174*4d7e907cSAndroid Build Coastguard Worker const char* address = event->device_info.audio_address;
175*4d7e907cSAndroid Build Coastguard Worker if (address != nullptr) {
176*4d7e907cSAndroid Build Coastguard Worker size_t size = strlen(address);
177*4d7e907cSAndroid Build Coastguard Worker if (size > tvInputEvent.deviceInfo.audioAddress.size()) {
178*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Audio address is too long. Address:" << address << "";
179*4d7e907cSAndroid Build Coastguard Worker return;
180*4d7e907cSAndroid Build Coastguard Worker }
181*4d7e907cSAndroid Build Coastguard Worker for (size_t i = 0; i < size; ++i) {
182*4d7e907cSAndroid Build Coastguard Worker tvInputEvent.deviceInfo.audioAddress[i] =
183*4d7e907cSAndroid Build Coastguard Worker static_cast<uint8_t>(event->device_info.audio_address[i]);
184*4d7e907cSAndroid Build Coastguard Worker }
185*4d7e907cSAndroid Build Coastguard Worker }
186*4d7e907cSAndroid Build Coastguard Worker mCallback->notify(tvInputEvent);
187*4d7e907cSAndroid Build Coastguard Worker }
188*4d7e907cSAndroid Build Coastguard Worker }
189*4d7e907cSAndroid Build Coastguard Worker
190*4d7e907cSAndroid Build Coastguard Worker // static
getSupportedConfigCount(uint32_t configCount,const tv_stream_config_t * configs)191*4d7e907cSAndroid Build Coastguard Worker uint32_t TvInput::getSupportedConfigCount(uint32_t configCount,
192*4d7e907cSAndroid Build Coastguard Worker const tv_stream_config_t* configs) {
193*4d7e907cSAndroid Build Coastguard Worker uint32_t supportedConfigCount = 0;
194*4d7e907cSAndroid Build Coastguard Worker for (uint32_t i = 0; i < configCount; ++i) {
195*4d7e907cSAndroid Build Coastguard Worker if (isSupportedStreamType(configs[i].type)) {
196*4d7e907cSAndroid Build Coastguard Worker supportedConfigCount++;
197*4d7e907cSAndroid Build Coastguard Worker }
198*4d7e907cSAndroid Build Coastguard Worker }
199*4d7e907cSAndroid Build Coastguard Worker return supportedConfigCount;
200*4d7e907cSAndroid Build Coastguard Worker }
201*4d7e907cSAndroid Build Coastguard Worker
202*4d7e907cSAndroid Build Coastguard Worker // static
isSupportedStreamType(int type)203*4d7e907cSAndroid Build Coastguard Worker bool TvInput::isSupportedStreamType(int type) {
204*4d7e907cSAndroid Build Coastguard Worker // Buffer producer type is no longer supported.
205*4d7e907cSAndroid Build Coastguard Worker return type != TV_STREAM_TYPE_BUFFER_PRODUCER;
206*4d7e907cSAndroid Build Coastguard Worker }
207*4d7e907cSAndroid Build Coastguard Worker
HIDL_FETCH_ITvInput(const char *)208*4d7e907cSAndroid Build Coastguard Worker ITvInput* HIDL_FETCH_ITvInput(const char* /* name */) {
209*4d7e907cSAndroid Build Coastguard Worker int ret = 0;
210*4d7e907cSAndroid Build Coastguard Worker const hw_module_t* hw_module = nullptr;
211*4d7e907cSAndroid Build Coastguard Worker tv_input_device_t* input_device;
212*4d7e907cSAndroid Build Coastguard Worker ret = hw_get_module(TV_INPUT_HARDWARE_MODULE_ID, &hw_module);
213*4d7e907cSAndroid Build Coastguard Worker if (ret == 0 && hw_module->methods->open != nullptr) {
214*4d7e907cSAndroid Build Coastguard Worker ret = hw_module->methods->open(hw_module, TV_INPUT_DEFAULT_DEVICE,
215*4d7e907cSAndroid Build Coastguard Worker reinterpret_cast<hw_device_t**>(&input_device));
216*4d7e907cSAndroid Build Coastguard Worker if (ret == 0) {
217*4d7e907cSAndroid Build Coastguard Worker return new TvInput(input_device);
218*4d7e907cSAndroid Build Coastguard Worker }
219*4d7e907cSAndroid Build Coastguard Worker else {
220*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "Passthrough failed to load legacy HAL.";
221*4d7e907cSAndroid Build Coastguard Worker return nullptr;
222*4d7e907cSAndroid Build Coastguard Worker }
223*4d7e907cSAndroid Build Coastguard Worker }
224*4d7e907cSAndroid Build Coastguard Worker else {
225*4d7e907cSAndroid Build Coastguard Worker LOG(ERROR) << "hw_get_module " << TV_INPUT_HARDWARE_MODULE_ID
226*4d7e907cSAndroid Build Coastguard Worker << " failed: " << ret;
227*4d7e907cSAndroid Build Coastguard Worker return nullptr;
228*4d7e907cSAndroid Build Coastguard Worker }
229*4d7e907cSAndroid Build Coastguard Worker }
230*4d7e907cSAndroid Build Coastguard Worker
231*4d7e907cSAndroid Build Coastguard Worker } // namespace implementation
232*4d7e907cSAndroid Build Coastguard Worker } // namespace V1_0
233*4d7e907cSAndroid Build Coastguard Worker } // namespace input
234*4d7e907cSAndroid Build Coastguard Worker } // namespace tv
235*4d7e907cSAndroid Build Coastguard Worker } // namespace hardware
236*4d7e907cSAndroid Build Coastguard Worker } // namespace android
237