blob: 525e223b2528815362cbe80608650087bb571884 [file] [log] [blame]
Jungshik Jang12e5dce2014-07-24 15:27:44 +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 Jange5a93372014-07-25 13:41:14 +090019import static android.hardware.hdmi.HdmiControlManager.TIMER_RECORDING_RESULT_EXTRA_CHECK_RECORDER_CONNECTION;
20import static android.hardware.hdmi.HdmiControlManager.TIMER_RECORDING_RESULT_EXTRA_FAIL_TO_RECORD_SELECTED_SOURCE;
Jungshik Jang12e5dce2014-07-24 15:27:44 +090021import static android.hardware.hdmi.HdmiControlManager.TIMER_RECORDING_TYPE_ANALOGUE;
22import static android.hardware.hdmi.HdmiControlManager.TIMER_RECORDING_TYPE_DIGITAL;
23import static android.hardware.hdmi.HdmiControlManager.TIMER_RECORDING_TYPE_EXTERNAL;
Jungshik Jang12e5dce2014-07-24 15:27:44 +090024
Donghyun Chobc6e3722016-11-04 05:25:52 +090025import android.hardware.tv.cec.V1_0.SendMessageResult;
Jungshik Jang12e5dce2014-07-24 15:27:44 +090026import android.util.Slog;
Jungshik Jang12e5dce2014-07-24 15:27:44 +090027import com.android.server.hdmi.HdmiControlService.SendMessageCallback;
Jungshik Jang12e5dce2014-07-24 15:27:44 +090028import java.util.Arrays;
29
30/**
31 * Feature action that performs timer recording.
32 */
Jungshik Jangb509c2e2014-08-07 13:45:01 +090033public class TimerRecordingAction extends HdmiCecFeatureAction {
Jungshik Jang12e5dce2014-07-24 15:27:44 +090034 private static final String TAG = "TimerRecordingAction";
35
36 // Timer out for waiting <Timer Status> 120s.
37 private static final int TIMER_STATUS_TIMEOUT_MS = 120000;
38
39 // State that waits for <Timer Status> once sending <Set XXX Timer>
40 private static final int STATE_WAITING_FOR_TIMER_STATUS = 1;
41
42 private final int mRecorderAddress;
43 private final int mSourceType;
44 private final byte[] mRecordSource;
45
46 TimerRecordingAction(HdmiCecLocalDevice source, int recorderAddress, int sourceType,
47 byte[] recordSource) {
48 super(source);
49 mRecorderAddress = recorderAddress;
50 mSourceType = sourceType;
51 mRecordSource = recordSource;
52 }
53
54 @Override
55 boolean start() {
56 sendTimerMessage();
57 return true;
58 }
59
60 private void sendTimerMessage() {
61 HdmiCecMessage message = null;
62 switch (mSourceType) {
63 case TIMER_RECORDING_TYPE_DIGITAL:
64 message = HdmiCecMessageBuilder.buildSetDigitalTimer(getSourceAddress(),
65 mRecorderAddress, mRecordSource);
66 break;
67 case TIMER_RECORDING_TYPE_ANALOGUE:
68 message = HdmiCecMessageBuilder.buildSetAnalogueTimer(getSourceAddress(),
69 mRecorderAddress, mRecordSource);
70 break;
71 case TIMER_RECORDING_TYPE_EXTERNAL:
72 message = HdmiCecMessageBuilder.buildSetExternalTimer(getSourceAddress(),
73 mRecorderAddress, mRecordSource);
74 break;
75 default:
Jungshik Jang326aef02014-11-05 12:50:35 +090076 tv().announceTimerRecordingResult(mRecorderAddress,
Jungshik Jange5a93372014-07-25 13:41:14 +090077 TIMER_RECORDING_RESULT_EXTRA_FAIL_TO_RECORD_SELECTED_SOURCE);
Jungshik Jang12e5dce2014-07-24 15:27:44 +090078 finish();
79 return;
80 }
81 sendCommand(message, new SendMessageCallback() {
82 @Override
83 public void onSendCompleted(int error) {
Donghyun Chobc6e3722016-11-04 05:25:52 +090084 if (error != SendMessageResult.SUCCESS) {
Jungshik Jang326aef02014-11-05 12:50:35 +090085 tv().announceTimerRecordingResult(mRecorderAddress,
Jungshik Jangfaa49bc2014-08-05 16:10:21 +090086 TIMER_RECORDING_RESULT_EXTRA_CHECK_RECORDER_CONNECTION);
Jungshik Jang12e5dce2014-07-24 15:27:44 +090087 finish();
88 return;
89 }
Jungshik Jangfaa49bc2014-08-05 16:10:21 +090090 mState = STATE_WAITING_FOR_TIMER_STATUS;
91 addTimer(mState, TIMER_STATUS_TIMEOUT_MS);
Jungshik Jang12e5dce2014-07-24 15:27:44 +090092 }
93 });
94 }
95
96 @Override
97 boolean processCommand(HdmiCecMessage cmd) {
Jungshik Jang5352081c2014-09-22 15:14:49 +090098 if (mState != STATE_WAITING_FOR_TIMER_STATUS
99 || cmd.getSource() != mRecorderAddress) {
Jungshik Jang12e5dce2014-07-24 15:27:44 +0900100 return false;
101 }
102
103 switch (cmd.getOpcode()) {
104 case Constants.MESSAGE_TIMER_STATUS:
105 return handleTimerStatus(cmd);
106 case Constants.MESSAGE_FEATURE_ABORT:
107 return handleFeatureAbort(cmd);
108 }
109 return false;
110 }
111
112 private boolean handleTimerStatus(HdmiCecMessage cmd) {
113 byte[] timerStatusData = cmd.getParams();
114 // [Timer Status Data] should be one or three bytes.
115 if (timerStatusData.length == 1 || timerStatusData.length == 3) {
Jungshik Jang326aef02014-11-05 12:50:35 +0900116 tv().announceTimerRecordingResult(mRecorderAddress, bytesToInt(timerStatusData));
Jungshik Jang12e5dce2014-07-24 15:27:44 +0900117 Slog.i(TAG, "Received [Timer Status Data]:" + Arrays.toString(timerStatusData));
118 } else {
119 Slog.w(TAG, "Invalid [Timer Status Data]:" + Arrays.toString(timerStatusData));
120 }
121
122 // Unlike one touch record, finish timer record when <Timer Status> is received.
123 finish();
124 return true;
125 }
126
127 private boolean handleFeatureAbort(HdmiCecMessage cmd) {
128 byte[] params = cmd.getParams();
Jungshik Jangfaa49bc2014-08-05 16:10:21 +0900129 int messageType = params[0] & 0xFF;
Jungshik Jang12e5dce2014-07-24 15:27:44 +0900130 switch (messageType) {
131 case Constants.MESSAGE_SET_DIGITAL_TIMER: // fall through
132 case Constants.MESSAGE_SET_ANALOG_TIMER: // fall through
133 case Constants.MESSAGE_SET_EXTERNAL_TIMER: // fall through
134 break;
135 default:
136 return false;
137 }
Jungshik Jangfaa49bc2014-08-05 16:10:21 +0900138 int reason = params[1] & 0xFF;
Jungshik Jang12e5dce2014-07-24 15:27:44 +0900139 Slog.i(TAG, "[Feature Abort] for " + messageType + " reason:" + reason);
Jungshik Jang326aef02014-11-05 12:50:35 +0900140 tv().announceTimerRecordingResult(mRecorderAddress,
141 TIMER_RECORDING_RESULT_EXTRA_CHECK_RECORDER_CONNECTION);
Jungshik Jang12e5dce2014-07-24 15:27:44 +0900142 finish();
143 return true;
144 }
145
146 // Convert byte array to int.
147 private static int bytesToInt(byte[] data) {
148 if (data.length > 4) {
149 throw new IllegalArgumentException("Invalid data size:" + Arrays.toString(data));
150 }
151 int result = 0;
152 for (int i = 0; i < data.length; ++i) {
153 int shift = (3 - i) * 8;
154 result |= ((data[i] & 0xFF) << shift);
155 }
156 return result;
157 }
158
159 @Override
160 void handleTimerEvent(int state) {
161 if (mState != state) {
162 Slog.w(TAG, "Timeout in invalid state:[Expected:" + mState + ", Actual:" + state + "]");
163 return;
164 }
165
Jungshik Jang326aef02014-11-05 12:50:35 +0900166 tv().announceTimerRecordingResult(mRecorderAddress,
167 TIMER_RECORDING_RESULT_EXTRA_CHECK_RECORDER_CONNECTION);
Jungshik Jang12e5dce2014-07-24 15:27:44 +0900168 finish();
169 }
170}