/*
|
* Copyright (C) 2014 The Android Open Source Project
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
|
package com.android.server.hdmi;
|
|
import android.util.FastImmutableArraySet;
|
import android.util.SparseArray;
|
|
/**
|
* Cache for incoming message. It caches {@link HdmiCecMessage} with source address and opcode
|
* as a key.
|
*
|
* <p>Note that whenever a device is removed it should call {@link #flushMessagesFrom(int)}
|
* to clean up messages come from the device.
|
*/
|
final class HdmiCecMessageCache {
|
private static final FastImmutableArraySet<Integer> CACHEABLE_OPCODES =
|
new FastImmutableArraySet<>(new Integer[] {
|
Constants.MESSAGE_SET_OSD_NAME,
|
Constants.MESSAGE_REPORT_PHYSICAL_ADDRESS,
|
Constants.MESSAGE_DEVICE_VENDOR_ID,
|
Constants.MESSAGE_CEC_VERSION,
|
});
|
|
// It's like [Source Logical Address, [Opcode, HdmiCecMessage]].
|
private final SparseArray<SparseArray<HdmiCecMessage>> mCache = new SparseArray<>();
|
|
HdmiCecMessageCache() {
|
}
|
|
/**
|
* Return a {@link HdmiCecMessage} corresponding to the given {@code address} and
|
* {@code opcode}.
|
*
|
* @param address a logical address of source device
|
* @param opcode opcode of message
|
* @return null if has no {@link HdmiCecMessage} matched to the given {@code address} and {code
|
* opcode}
|
*/
|
public HdmiCecMessage getMessage(int address, int opcode) {
|
SparseArray<HdmiCecMessage> messages = mCache.get(address);
|
if (messages == null) {
|
return null;
|
}
|
|
return messages.get(opcode);
|
}
|
|
/**
|
* Flush all {@link HdmiCecMessage}s sent from the given {@code address}.
|
*
|
* @param address a logical address of source device
|
*/
|
public void flushMessagesFrom(int address) {
|
mCache.remove(address);
|
}
|
|
/**
|
* Flush all cached {@link HdmiCecMessage}s.
|
*/
|
public void flushAll() {
|
mCache.clear();
|
}
|
|
/**
|
* Cache incoming {@link HdmiCecMessage}. If opcode of message is not listed on
|
* cacheable opcodes list, just ignore it.
|
*
|
* @param message a {@link HdmiCecMessage} to be cached
|
*/
|
public void cacheMessage(HdmiCecMessage message) {
|
int opcode = message.getOpcode();
|
if (!isCacheable(opcode)) {
|
return;
|
}
|
|
int source = message.getSource();
|
SparseArray<HdmiCecMessage> messages = mCache.get(source);
|
if (messages == null) {
|
messages = new SparseArray<>();
|
mCache.put(source, messages);
|
}
|
messages.put(opcode, message);
|
}
|
|
private boolean isCacheable(int opcode) {
|
return CACHEABLE_OPCODES.contains(opcode);
|
}
|
}
|