Home | History | Annotate | Download | only in hdmi
      1 /*
      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 
     17 package com.android.server.hdmi;
     18 
     19 import android.hardware.hdmi.HdmiDeviceInfo;
     20 import android.util.SparseArray;
     21 
     22 /**
     23  * A helper class to validates {@link HdmiCecMessage}.
     24  */
     25 public final class HdmiCecMessageValidator {
     26     private static final String TAG = "HdmiCecMessageValidator";
     27 
     28     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;
     32     static final int ERROR_PARAMETER_SHORT = 4;
     33 
     34     private final HdmiControlService mService;
     35 
     36     interface ParameterValidator {
     37         /**
     38          * @return errorCode errorCode can be {@link #OK}, {@link #ERROR_PARAMETER} or
     39          *         {@link #ERROR_PARAMETER_SHORT}.
     40          */
     41         int isValid(byte[] params);
     42     }
     43 
     44     // 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<>();
     64 
     65     public HdmiCecMessageValidator(HdmiControlService service) {
     66         mService = service;
     67 
     68         // Messages related to the physical address.
     69         PhysicalAddressValidator physicalAddressValidator = new PhysicalAddressValidator();
     70         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,
     82                 new SystemAudioModeRequestValidator(), DEST_DIRECT);
     83 
     84         // Messages have no parameter.
     85         FixedLengthValidator noneValidator = new FixedLengthValidator(0);
     86         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);
    116 
    117         // TODO: Validate more than length for the following messages.
    118 
    119         // Messages for the One Touch Record.
    120         FixedLengthValidator oneByteValidator = new FixedLengthValidator(1);
    121         addValidationInfo(Constants.MESSAGE_RECORD_ON,
    122                 new VariableLengthValidator(1, 8), DEST_DIRECT);
    123         addValidationInfo(Constants.MESSAGE_RECORD_STATUS, oneByteValidator, DEST_DIRECT);
    124 
    125         // TODO: Handle messages for the Timer Programming.
    126 
    127         // Messages for the System Information.
    128         addValidationInfo(Constants.MESSAGE_CEC_VERSION, oneByteValidator, DEST_DIRECT);
    129         addValidationInfo(Constants.MESSAGE_SET_MENU_LANGUAGE,
    130                 new FixedLengthValidator(3), DEST_BROADCAST);
    131 
    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);
    138         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,
    143                 new VariableLengthValidator(1, 14), DEST_DIRECT | SRC_UNREGISTERED);
    144         addValidationInfo(Constants.MESSAGE_VENDOR_COMMAND_WITH_ID,
    145                 new VariableLengthValidator(4, 14), DEST_ALL | SRC_UNREGISTERED);
    146         addValidationInfo(Constants.MESSAGE_VENDOR_REMOTE_BUTTON_DOWN,
    147                 maxLengthValidator, DEST_ALL | SRC_UNREGISTERED);
    148 
    149         // Messages for the OSD.
    150         addValidationInfo(Constants.MESSAGE_SET_OSD_STRING, maxLengthValidator, DEST_DIRECT);
    151         addValidationInfo(Constants.MESSAGE_SET_OSD_NAME, maxLengthValidator, DEST_DIRECT);
    152 
    153         // Messages for the Device Menu Control.
    154         addValidationInfo(Constants.MESSAGE_MENU_REQUEST, oneByteValidator, DEST_DIRECT);
    155         addValidationInfo(Constants.MESSAGE_MENU_STATUS, oneByteValidator, DEST_DIRECT);
    156 
    157         // Messages for the Remote Control Passthrough.
    158         // TODO: Parse the first parameter and determine if it can have the next parameter.
    159         addValidationInfo(Constants.MESSAGE_USER_CONTROL_PRESSED,
    160                 new VariableLengthValidator(1, 2), DEST_DIRECT);
    161 
    162         // Messages for the Power Status.
    163         addValidationInfo(Constants.MESSAGE_REPORT_POWER_STATUS, oneByteValidator, DEST_DIRECT);
    164 
    165         // Messages for the General Protocol.
    166         addValidationInfo(Constants.MESSAGE_FEATURE_ABORT,
    167                 new FixedLengthValidator(2), DEST_DIRECT);
    168 
    169         // Messages for the System Audio Control.
    170         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);
    178 
    179         // Messages for the Audio Rate Control.
    180         addValidationInfo(Constants.MESSAGE_SET_AUDIO_RATE, oneByteValidator, DEST_DIRECT);
    181 
    182         // All Messages for the ARC have no parameters.
    183 
    184         // Messages for the Capability Discovery and Control.
    185         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));
    191     }
    192 
    193     int isValid(HdmiCecMessage message) {
    194         int opcode = message.getOpcode();
    195         ValidationInfo info = mValidationInfo.get(opcode);
    196         if (info == null) {
    197             HdmiLogger.warning("No validation information for the message: " + message);
    198             return OK;
    199         }
    200 
    201         // Check the source field.
    202         if (message.getSource() == Constants.ADDR_UNREGISTERED &&
    203                 (info.addressType & SRC_UNREGISTERED) == 0) {
    204             HdmiLogger.warning("Unexpected source: " + message);
    205             return ERROR_SOURCE;
    206         }
    207         // Check the destination field.
    208         if (message.getDestination() == Constants.ADDR_BROADCAST) {
    209             if ((info.addressType & DEST_BROADCAST) == 0) {
    210                 HdmiLogger.warning("Unexpected broadcast message: " + message);
    211                 return ERROR_DESTINATION;
    212             }
    213         } else {  // Direct addressing.
    214             if ((info.addressType & DEST_DIRECT) == 0) {
    215                 HdmiLogger.warning("Unexpected direct message: " + message);
    216                 return ERROR_DESTINATION;
    217             }
    218         }
    219 
    220         // Check the parameter type.
    221         int errorCode = info.parameterValidator.isValid(message.getParams());
    222         if (errorCode != OK) {
    223             HdmiLogger.warning("Unexpected parameters: " + message);
    224             return errorCode;
    225         }
    226         return OK;
    227     }
    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
    237         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;
    241         }
    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
    254         public int isValid(byte[] params) {
    255             return params.length < mMinLength ? ERROR_PARAMETER_SHORT : OK;
    256         }
    257     }
    258 
    259     private boolean isValidPhysicalAddress(byte[] params, int offset) {
    260         // 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         }
    266         int path = HdmiUtils.twoBytesToInt(params, offset);
    267         if (path != Constants.INVALID_PHYSICAL_ADDRESS && path == mService.getPhysicalAddress()) {
    268             return true;
    269         }
    270         int portId = mService.pathToPortId(path);
    271         if (portId == Constants.INVALID_PORT_ID) {
    272             return false;
    273         }
    274         return true;
    275     }
    276 
    277     /**
    278      * Check if the given type is valid. A valid type is one of the actual logical device types
    279      * defined in the standard ({@link HdmiDeviceInfo#DEVICE_TV},
    280      * {@link HdmiDeviceInfo#DEVICE_PLAYBACK}, {@link HdmiDeviceInfo#DEVICE_TUNER},
    281      * {@link HdmiDeviceInfo#DEVICE_RECORDER}, and {@link HdmiDeviceInfo#DEVICE_AUDIO_SYSTEM}).
    282      *
    283      * @param type device type
    284      * @return true if the given type is valid
    285      */
    286     static boolean isValidType(int type) {
    287         return (HdmiDeviceInfo.DEVICE_TV <= type
    288                 && type <= HdmiDeviceInfo.DEVICE_VIDEO_PROCESSOR)
    289                 && type != HdmiDeviceInfo.DEVICE_RESERVED;
    290     }
    291 
    292     private static int toErrorCode(boolean success) {
    293         return success ? OK : ERROR_PARAMETER;
    294     }
    295 
    296     private class PhysicalAddressValidator implements ParameterValidator {
    297         @Override
    298         public int isValid(byte[] params) {
    299             if (params.length < 2) {
    300                 return ERROR_PARAMETER_SHORT;
    301             }
    302             return toErrorCode(isValidPhysicalAddress(params, 0));
    303         }
    304     }
    305 
    306     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 
    317     private class ReportPhysicalAddressValidator implements ParameterValidator {
    318         @Override
    319         public int isValid(byte[] params) {
    320             if (params.length < 3) {
    321                 return ERROR_PARAMETER_SHORT;
    322             }
    323             return toErrorCode(isValidPhysicalAddress(params, 0) && isValidType(params[2]));
    324         }
    325     }
    326 
    327     private class RoutingChangeValidator implements ParameterValidator {
    328         @Override
    329         public int isValid(byte[] params) {
    330             if (params.length < 4) {
    331                 return ERROR_PARAMETER_SHORT;
    332             }
    333             return toErrorCode(
    334                     isValidPhysicalAddress(params, 0) && isValidPhysicalAddress(params, 2));
    335         }
    336     }
    337 }
    338