blob: 4ad51de2e25b46740f48d4f525a5e378b20b43b4 [file] [log] [blame]
Yuncheol Heo75a77e72014-07-09 18:27:53 +09001/*
2 * Copyright (C) 2014 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
17package com.android.server.hdmi;
18
Jungshik Jang61f4fbd2014-08-06 19:21:12 +090019import android.hardware.hdmi.HdmiDeviceInfo;
Yuncheol Heo75a77e72014-07-09 18:27:53 +090020import android.util.SparseArray;
21
22/**
23 * A helper class to validates {@link HdmiCecMessage}.
24 */
Amy1d0b1372018-05-24 14:36:25 -070025public class HdmiCecMessageValidator {
Yuncheol Heo75a77e72014-07-09 18:27:53 +090026 private static final String TAG = "HdmiCecMessageValidator";
27
Yuncheol Heo4c212892014-09-12 14:32:46 +090028 static final int OK = 0;
29 static final int ERROR_SOURCE = 1;
30 static final int ERROR_DESTINATION = 2;
31 static final int ERROR_PARAMETER = 3;
Yuncheol Heoa95f1a92014-11-06 08:25:39 +090032 static final int ERROR_PARAMETER_SHORT = 4;
Yuncheol Heo4c212892014-09-12 14:32:46 +090033
Yuncheol Heo75a77e72014-07-09 18:27:53 +090034 private final HdmiControlService mService;
35
36 interface ParameterValidator {
Yuncheol Heoa95f1a92014-11-06 08:25:39 +090037 /**
38 * @return errorCode errorCode can be {@link #OK}, {@link #ERROR_PARAMETER} or
39 * {@link #ERROR_PARAMETER_SHORT}.
40 */
41 int isValid(byte[] params);
Yuncheol Heo75a77e72014-07-09 18:27:53 +090042 }
43
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +090044 // Only the direct addressing is allowed.
45 private static final int DEST_DIRECT = 1 << 0;
46 // Only the broadcast addressing is allowed.
47 private static final int DEST_BROADCAST = 1 << 1;
48 // Both the direct and the broadcast addressing are allowed.
49 private static final int DEST_ALL = DEST_DIRECT | DEST_BROADCAST;
50 // True if the messages from address 15 (unregistered) are allowed.
51 private static final int SRC_UNREGISTERED = 1 << 2;
52
53 private static class ValidationInfo {
54 public final ParameterValidator parameterValidator;
55 public final int addressType;
56
57 public ValidationInfo(ParameterValidator validator, int type) {
58 parameterValidator = validator;
59 addressType = type;
60 }
61 }
62
63 final SparseArray<ValidationInfo> mValidationInfo = new SparseArray<>();
Yuncheol Heo75a77e72014-07-09 18:27:53 +090064
65 public HdmiCecMessageValidator(HdmiControlService service) {
66 mService = service;
67
68 // Messages related to the physical address.
69 PhysicalAddressValidator physicalAddressValidator = new PhysicalAddressValidator();
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +090070 addValidationInfo(Constants.MESSAGE_ACTIVE_SOURCE,
71 physicalAddressValidator, DEST_BROADCAST | SRC_UNREGISTERED);
72 addValidationInfo(Constants.MESSAGE_INACTIVE_SOURCE, physicalAddressValidator, DEST_DIRECT);
73 addValidationInfo(Constants.MESSAGE_REPORT_PHYSICAL_ADDRESS,
74 new ReportPhysicalAddressValidator(), DEST_BROADCAST | SRC_UNREGISTERED);
75 addValidationInfo(Constants.MESSAGE_ROUTING_CHANGE,
76 new RoutingChangeValidator(), DEST_BROADCAST | SRC_UNREGISTERED);
77 addValidationInfo(Constants.MESSAGE_ROUTING_INFORMATION,
78 physicalAddressValidator, DEST_BROADCAST | SRC_UNREGISTERED);
79 addValidationInfo(Constants.MESSAGE_SET_STREAM_PATH,
80 physicalAddressValidator, DEST_BROADCAST);
81 addValidationInfo(Constants.MESSAGE_SYSTEM_AUDIO_MODE_REQUEST,
Yuncheol Heo03611472014-11-06 17:21:04 +090082 new SystemAudioModeRequestValidator(), DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +090083
84 // Messages have no parameter.
85 FixedLengthValidator noneValidator = new FixedLengthValidator(0);
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +090086 addValidationInfo(Constants.MESSAGE_ABORT, noneValidator, DEST_DIRECT);
87 addValidationInfo(Constants.MESSAGE_GET_CEC_VERSION, noneValidator, DEST_DIRECT);
88 addValidationInfo(Constants.MESSAGE_GET_MENU_LANGUAGE,
89 noneValidator, DEST_DIRECT | SRC_UNREGISTERED);
90 addValidationInfo(Constants.MESSAGE_GIVE_AUDIO_STATUS, noneValidator, DEST_DIRECT);
91 addValidationInfo(Constants.MESSAGE_GIVE_DEVICE_POWER_STATUS, noneValidator, DEST_DIRECT);
92 addValidationInfo(Constants.MESSAGE_GIVE_DEVICE_VENDOR_ID,
93 noneValidator, DEST_DIRECT | SRC_UNREGISTERED);
94 addValidationInfo(Constants.MESSAGE_GIVE_OSD_NAME, noneValidator, DEST_DIRECT);
95 addValidationInfo(Constants.MESSAGE_GIVE_PHYSICAL_ADDRESS,
96 noneValidator, DEST_DIRECT | SRC_UNREGISTERED);
97 addValidationInfo(Constants.MESSAGE_GIVE_SYSTEM_AUDIO_MODE_STATUS,
98 noneValidator, DEST_DIRECT);
99 addValidationInfo(Constants.MESSAGE_IMAGE_VIEW_ON, noneValidator, DEST_DIRECT);
100 addValidationInfo(Constants.MESSAGE_INITIATE_ARC, noneValidator, DEST_DIRECT);
101 addValidationInfo(Constants.MESSAGE_RECORD_OFF, noneValidator, DEST_DIRECT);
102 addValidationInfo(Constants.MESSAGE_RECORD_TV_SCREEN, noneValidator, DEST_DIRECT);
103 addValidationInfo(Constants.MESSAGE_REPORT_ARC_INITIATED, noneValidator, DEST_DIRECT);
104 addValidationInfo(Constants.MESSAGE_REPORT_ARC_TERMINATED, noneValidator, DEST_DIRECT);
105 addValidationInfo(Constants.MESSAGE_REQUEST_ARC_INITIATION, noneValidator, DEST_DIRECT);
106 addValidationInfo(Constants.MESSAGE_REQUEST_ARC_TERMINATION, noneValidator, DEST_DIRECT);
107 addValidationInfo(Constants.MESSAGE_REQUEST_ACTIVE_SOURCE,
108 noneValidator, DEST_BROADCAST | SRC_UNREGISTERED);
109 addValidationInfo(Constants.MESSAGE_STANDBY, noneValidator, DEST_ALL | SRC_UNREGISTERED);
110 addValidationInfo(Constants.MESSAGE_TERMINATE_ARC, noneValidator, DEST_DIRECT);
111 addValidationInfo(Constants.MESSAGE_TEXT_VIEW_ON, noneValidator, DEST_DIRECT);
112 addValidationInfo(Constants.MESSAGE_TUNER_STEP_DECREMENT, noneValidator, DEST_DIRECT);
113 addValidationInfo(Constants.MESSAGE_TUNER_STEP_INCREMENT, noneValidator, DEST_DIRECT);
114 addValidationInfo(Constants.MESSAGE_USER_CONTROL_RELEASED, noneValidator, DEST_DIRECT);
115 addValidationInfo(Constants.MESSAGE_VENDOR_REMOTE_BUTTON_UP, noneValidator, DEST_ALL);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900116
117 // TODO: Validate more than length for the following messages.
118
119 // Messages for the One Touch Record.
120 FixedLengthValidator oneByteValidator = new FixedLengthValidator(1);
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900121 addValidationInfo(Constants.MESSAGE_RECORD_ON,
122 new VariableLengthValidator(1, 8), DEST_DIRECT);
123 addValidationInfo(Constants.MESSAGE_RECORD_STATUS, oneByteValidator, DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900124
125 // TODO: Handle messages for the Timer Programming.
126
127 // Messages for the System Information.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900128 addValidationInfo(Constants.MESSAGE_CEC_VERSION, oneByteValidator, DEST_DIRECT);
129 addValidationInfo(Constants.MESSAGE_SET_MENU_LANGUAGE,
130 new FixedLengthValidator(3), DEST_BROADCAST);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900131
132 // TODO: Handle messages for the Deck Control.
133
134 // TODO: Handle messages for the Tuner Control.
135
136 // Messages for the Vendor Specific Commands.
137 VariableLengthValidator maxLengthValidator = new VariableLengthValidator(0, 14);
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900138 addValidationInfo(Constants.MESSAGE_DEVICE_VENDOR_ID,
139 new FixedLengthValidator(3), DEST_BROADCAST);
140 // Allow unregistered source for all vendor specific commands, because we don't know
141 // how to use the commands at this moment.
142 addValidationInfo(Constants.MESSAGE_VENDOR_COMMAND,
Jinsuk Kimb96bccf2015-10-16 15:15:18 +0900143 new VariableLengthValidator(1, 14), DEST_DIRECT | SRC_UNREGISTERED);
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900144 addValidationInfo(Constants.MESSAGE_VENDOR_COMMAND_WITH_ID,
Jinsuk Kima95794b2014-12-20 08:43:27 +0900145 new VariableLengthValidator(4, 14), DEST_ALL | SRC_UNREGISTERED);
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900146 addValidationInfo(Constants.MESSAGE_VENDOR_REMOTE_BUTTON_DOWN,
147 maxLengthValidator, DEST_ALL | SRC_UNREGISTERED);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900148
149 // Messages for the OSD.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900150 addValidationInfo(Constants.MESSAGE_SET_OSD_STRING, maxLengthValidator, DEST_DIRECT);
151 addValidationInfo(Constants.MESSAGE_SET_OSD_NAME, maxLengthValidator, DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900152
Yuncheol Heo184b1242014-09-12 15:09:07 +0900153 // Messages for the Device Menu Control.
154 addValidationInfo(Constants.MESSAGE_MENU_REQUEST, oneByteValidator, DEST_DIRECT);
155 addValidationInfo(Constants.MESSAGE_MENU_STATUS, oneByteValidator, DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900156
157 // Messages for the Remote Control Passthrough.
158 // TODO: Parse the first parameter and determine if it can have the next parameter.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900159 addValidationInfo(Constants.MESSAGE_USER_CONTROL_PRESSED,
160 new VariableLengthValidator(1, 2), DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900161
162 // Messages for the Power Status.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900163 addValidationInfo(Constants.MESSAGE_REPORT_POWER_STATUS, oneByteValidator, DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900164
165 // Messages for the General Protocol.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900166 addValidationInfo(Constants.MESSAGE_FEATURE_ABORT,
167 new FixedLengthValidator(2), DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900168
169 // Messages for the System Audio Control.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900170 addValidationInfo(Constants.MESSAGE_REPORT_AUDIO_STATUS, oneByteValidator, DEST_DIRECT);
171 addValidationInfo(Constants.MESSAGE_REPORT_SHORT_AUDIO_DESCRIPTOR,
172 new FixedLengthValidator(3), DEST_DIRECT);
173 addValidationInfo(Constants.MESSAGE_REQUEST_SHORT_AUDIO_DESCRIPTOR,
174 oneByteValidator, DEST_DIRECT);
175 addValidationInfo(Constants.MESSAGE_SET_SYSTEM_AUDIO_MODE, oneByteValidator, DEST_ALL);
176 addValidationInfo(Constants.MESSAGE_SYSTEM_AUDIO_MODE_STATUS,
177 oneByteValidator, DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900178
179 // Messages for the Audio Rate Control.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900180 addValidationInfo(Constants.MESSAGE_SET_AUDIO_RATE, oneByteValidator, DEST_DIRECT);
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900181
182 // All Messages for the ARC have no parameters.
183
184 // Messages for the Capability Discovery and Control.
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900185 addValidationInfo(Constants.MESSAGE_CDC_MESSAGE, maxLengthValidator,
186 DEST_BROADCAST | SRC_UNREGISTERED);
187 }
188
189 private void addValidationInfo(int opcode, ParameterValidator validator, int addrType) {
190 mValidationInfo.append(opcode, new ValidationInfo(validator, addrType));
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900191 }
192
Yuncheol Heo4c212892014-09-12 14:32:46 +0900193 int isValid(HdmiCecMessage message) {
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900194 int opcode = message.getOpcode();
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900195 ValidationInfo info = mValidationInfo.get(opcode);
196 if (info == null) {
Jungshik Jang2e8f1b62014-09-03 08:28:02 +0900197 HdmiLogger.warning("No validation information for the message: " + message);
Yuncheol Heo4c212892014-09-12 14:32:46 +0900198 return OK;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900199 }
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900200
201 // Check the source field.
202 if (message.getSource() == Constants.ADDR_UNREGISTERED &&
203 (info.addressType & SRC_UNREGISTERED) == 0) {
Jungshik Jang2e8f1b62014-09-03 08:28:02 +0900204 HdmiLogger.warning("Unexpected source: " + message);
Yuncheol Heo4c212892014-09-12 14:32:46 +0900205 return ERROR_SOURCE;
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900206 }
207 // Check the destination field.
208 if (message.getDestination() == Constants.ADDR_BROADCAST) {
209 if ((info.addressType & DEST_BROADCAST) == 0) {
Jungshik Jang2e8f1b62014-09-03 08:28:02 +0900210 HdmiLogger.warning("Unexpected broadcast message: " + message);
Yuncheol Heo4c212892014-09-12 14:32:46 +0900211 return ERROR_DESTINATION;
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900212 }
213 } else { // Direct addressing.
214 if ((info.addressType & DEST_DIRECT) == 0) {
Jungshik Jang2e8f1b62014-09-03 08:28:02 +0900215 HdmiLogger.warning("Unexpected direct message: " + message);
Yuncheol Heo4c212892014-09-12 14:32:46 +0900216 return ERROR_DESTINATION;
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900217 }
218 }
219
220 // Check the parameter type.
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900221 int errorCode = info.parameterValidator.isValid(message.getParams());
222 if (errorCode != OK) {
Jungshik Jang2e8f1b62014-09-03 08:28:02 +0900223 HdmiLogger.warning("Unexpected parameters: " + message);
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900224 return errorCode;
Yuncheol Heoe9b9b1e2014-07-10 18:52:28 +0900225 }
Yuncheol Heo4c212892014-09-12 14:32:46 +0900226 return OK;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900227 }
228
229 private static class FixedLengthValidator implements ParameterValidator {
230 private final int mLength;
231
232 public FixedLengthValidator(int length) {
233 mLength = length;
234 }
235
236 @Override
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900237 public int isValid(byte[] params) {
238 // If the length is longer than expected, we assume it's OK since the parameter can be
239 // extended in the future version.
240 return params.length < mLength ? ERROR_PARAMETER_SHORT : OK;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900241 }
242 }
243
244 private static class VariableLengthValidator implements ParameterValidator {
245 private final int mMinLength;
246 private final int mMaxLength;
247
248 public VariableLengthValidator(int minLength, int maxLength) {
249 mMinLength = minLength;
250 mMaxLength = maxLength;
251 }
252
253 @Override
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900254 public int isValid(byte[] params) {
255 return params.length < mMinLength ? ERROR_PARAMETER_SHORT : OK;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900256 }
257 }
258
259 private boolean isValidPhysicalAddress(byte[] params, int offset) {
Yuncheol Heoe946ed82014-07-25 14:05:19 +0900260 // TODO: Add more logic like validating 1.0.1.0.
261
262 if (!mService.isTvDevice()) {
263 // If the device is not TV, we can't convert path to port-id, so stop here.
264 return true;
265 }
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900266 int path = HdmiUtils.twoBytesToInt(params, offset);
Yuncheol Heo7dea98f2014-08-07 17:58:59 +0900267 if (path != Constants.INVALID_PHYSICAL_ADDRESS && path == mService.getPhysicalAddress()) {
268 return true;
269 }
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900270 int portId = mService.pathToPortId(path);
271 if (portId == Constants.INVALID_PORT_ID) {
272 return false;
273 }
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900274 return true;
275 }
276
277 /**
Jungshik Jang8e93c842014-08-06 15:48:33 +0900278 * Check if the given type is valid. A valid type is one of the actual logical device types
Yuncheol Heo7dea98f2014-08-07 17:58:59 +0900279 * defined in the standard ({@link HdmiDeviceInfo#DEVICE_TV},
280 * {@link HdmiDeviceInfo#DEVICE_PLAYBACK}, {@link HdmiDeviceInfo#DEVICE_TUNER},
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900281 * {@link HdmiDeviceInfo#DEVICE_RECORDER}, and {@link HdmiDeviceInfo#DEVICE_AUDIO_SYSTEM}).
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900282 *
283 * @param type device type
284 * @return true if the given type is valid
285 */
286 static boolean isValidType(int type) {
Jungshik Jang61f4fbd2014-08-06 19:21:12 +0900287 return (HdmiDeviceInfo.DEVICE_TV <= type
288 && type <= HdmiDeviceInfo.DEVICE_VIDEO_PROCESSOR)
289 && type != HdmiDeviceInfo.DEVICE_RESERVED;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900290 }
291
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900292 private static int toErrorCode(boolean success) {
293 return success ? OK : ERROR_PARAMETER;
294 }
295
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900296 private class PhysicalAddressValidator implements ParameterValidator {
297 @Override
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900298 public int isValid(byte[] params) {
299 if (params.length < 2) {
300 return ERROR_PARAMETER_SHORT;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900301 }
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900302 return toErrorCode(isValidPhysicalAddress(params, 0));
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900303 }
304 }
305
Yuncheol Heo03611472014-11-06 17:21:04 +0900306 private class SystemAudioModeRequestValidator extends PhysicalAddressValidator {
307 @Override
308 public int isValid(byte[] params) {
309 // TV can send <System Audio Mode Request> with no parameters to terminate system audio.
310 if (params.length == 0) {
311 return OK;
312 }
313 return super.isValid(params);
314 }
315 }
316
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900317 private class ReportPhysicalAddressValidator implements ParameterValidator {
318 @Override
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900319 public int isValid(byte[] params) {
320 if (params.length < 3) {
321 return ERROR_PARAMETER_SHORT;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900322 }
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900323 return toErrorCode(isValidPhysicalAddress(params, 0) && isValidType(params[2]));
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900324 }
325 }
326
327 private class RoutingChangeValidator implements ParameterValidator {
328 @Override
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900329 public int isValid(byte[] params) {
330 if (params.length < 4) {
331 return ERROR_PARAMETER_SHORT;
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900332 }
Yuncheol Heoa95f1a92014-11-06 08:25:39 +0900333 return toErrorCode(
334 isValidPhysicalAddress(params, 0) && isValidPhysicalAddress(params, 2));
Yuncheol Heo75a77e72014-07-09 18:27:53 +0900335 }
336 }
337}