WifiDisplaySource.cpp revision 90a92053219ae50ddf4bb54e3d54db2d309e2b8d
1d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber/* 2d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * Copyright 2012, The Android Open Source Project 3d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * 4d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * Licensed under the Apache License, Version 2.0 (the "License"); 5d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * you may not use this file except in compliance with the License. 6d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * You may obtain a copy of the License at 7d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * 8d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * http://www.apache.org/licenses/LICENSE-2.0 9d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * 10d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * Unless required by applicable law or agreed to in writing, software 11d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * distributed under the License is distributed on an "AS IS" BASIS, 12d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * See the License for the specific language governing permissions and 14d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber * limitations under the License. 15d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber */ 16d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1772f6aea5afba3ff8ab7e8eab49552d65ee3bb97bAndreas Huber//#define LOG_NDEBUG 0 18d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#define LOG_TAG "WifiDisplaySource" 19d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <utils/Log.h> 20d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 21d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include "WifiDisplaySource.h" 22d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include "PlaybackSession.h" 23b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#include "Parameters.h" 24d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include "ParsedMessage.h" 2590a92053219ae50ddf4bb54e3d54db2d309e2b8dAndreas Huber#include "Sender.h" 26d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 27b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#include <binder/IServiceManager.h> 280b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber#include <gui/ISurfaceTexture.h> 29b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#include <media/IHDCP.h> 30b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#include <media/IMediaPlayerService.h> 310b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber#include <media/IRemoteDisplayClient.h> 32d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/ABuffer.h> 33d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/ADebug.h> 34d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/AMessage.h> 35d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/MediaErrors.h> 36d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 37bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber#include <arpa/inet.h> 38bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber#include <cutils/properties.h> 39bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 40d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber#include <ctype.h> 41d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 42d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubernamespace android { 43d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 440b73d4730202fcad53aefc4314a06e7b95f442f0Andreas HuberWifiDisplaySource::WifiDisplaySource( 450b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber const sp<ANetworkSession> &netSession, 460b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber const sp<IRemoteDisplayClient> &client) 47ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber : mState(INITIALIZED), 48ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mNetSession(netSession), 490b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber mClient(client), 50d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mSessionID(0), 51ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber mStopReplyID(0), 52d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber mChosenRTPPort(-1), 53e7bd24af08ef0722fb124a550662bcec48c56f86Andreas Huber mUsingPCMAudio(false), 54c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientSessionID(0), 55d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mReaperPending(false), 560328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mNextCSeq(1), 570328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mUsingHDCP(false), 580328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mIsHDCP2_0(false), 590328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mHDCPPort(0), 600328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mHDCPInitializationComplete(false), 610328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mSetupTriggerDeferred(false) 62b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber{ 63d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 64d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 65d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas HuberWifiDisplaySource::~WifiDisplaySource() { 66d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 67d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 68bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huberstatus_t WifiDisplaySource::start(const char *iface) { 69ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber CHECK_EQ(mState, INITIALIZED); 70ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 71d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<AMessage> msg = new AMessage(kWhatStart, id()); 72bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber msg->setString("iface", iface); 73d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 74d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<AMessage> response; 75d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = msg->postAndAwaitResponse(&response); 76d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 77d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 78d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 79d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 80d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 81d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!response->findInt32("err", &err)) { 82d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber err = OK; 83d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 84d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 85d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 86d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 87d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 88d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::stop() { 89d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<AMessage> msg = new AMessage(kWhatStop, id()); 90d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 91d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<AMessage> response; 92d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = msg->postAndAwaitResponse(&response); 93d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 94d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 95d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 96d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 97d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 98d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!response->findInt32("err", &err)) { 99d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber err = OK; 100d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 101d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 102d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 103d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 104d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 105d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onMessageReceived(const sp<AMessage> &msg) { 106d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber switch (msg->what()) { 107d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case kWhatStart: 108d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 109d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber uint32_t replyID; 110d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->senderAwaitsResponse(&replyID)); 111d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 112bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber AString iface; 113bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber CHECK(msg->findString("iface", &iface)); 114bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 115bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber status_t err = OK; 116bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 117bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber ssize_t colonPos = iface.find(":"); 118bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 119bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber unsigned long port; 120bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 121bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber if (colonPos >= 0) { 122bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber const char *s = iface.c_str() + colonPos + 1; 123bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 124bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber char *end; 125bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber port = strtoul(s, &end, 10); 126bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 127bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber if (end == s || *end != '\0' || port > 65535) { 128bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber err = -EINVAL; 129bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber } else { 130bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber iface.erase(colonPos, iface.size() - colonPos); 131bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber } 132bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber } else { 133bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber port = kWifiDisplayDefaultPort; 134bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber } 135bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber 136bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber if (err == OK) { 137bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber if (inet_aton(iface.c_str(), &mInterfaceAddr) != 0) { 138bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber sp<AMessage> notify = new AMessage(kWhatRTSPNotify, id()); 139d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 140bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber err = mNetSession->createRTSPServer( 141bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber mInterfaceAddr, port, notify, &mSessionID); 142bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber } else { 143bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber err = -EINVAL; 144bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber } 145bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber } 146d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 147ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber if (err == OK) { 148ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = AWAITING_CLIENT_CONNECTION; 149ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber } 150ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 151d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<AMessage> response = new AMessage; 152d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->setInt32("err", err); 153d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->postReply(replyID); 154d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 155d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 156d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 157d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case kWhatRTSPNotify: 158d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 159d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t reason; 160d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findInt32("reason", &reason)); 161d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 162d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber switch (reason) { 163d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case ANetworkSession::kWhatError: 164d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 165d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID; 166d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findInt32("sessionID", &sessionID)); 167d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 168d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t err; 169d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findInt32("err", &err)); 170d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 171d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString detail; 172d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findString("detail", &detail)); 173d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 174d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ALOGE("An error occurred in session %d (%d, '%s/%s').", 175d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sessionID, 176d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber err, 177d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber detail.c_str(), 178d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber strerror(-err)); 179d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 180d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mNetSession->destroySession(sessionID); 181d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 182c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (sessionID == mClientSessionID) { 183ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClientSessionID = 0; 184c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 185ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClient->onDisplayError( 186ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber IRemoteDisplayClient::kDisplayErrorUnknown); 187c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } 188d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 189d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 190d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 191d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case ANetworkSession::kWhatClientConnected: 192d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 193d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID; 194d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findInt32("sessionID", &sessionID)); 195d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 196c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClientSessionID > 0) { 197c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber ALOGW("A client tried to connect, but we already " 198c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber "have one."); 199d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 200c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mNetSession->destroySession(sessionID); 201c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber break; 202c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } 203c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 204ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber CHECK_EQ(mState, AWAITING_CLIENT_CONNECTION); 205ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 206c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK(msg->findString("client-ip", &mClientInfo.mRemoteIP)); 207c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK(msg->findString("server-ip", &mClientInfo.mLocalIP)); 208c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 209c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClientInfo.mRemoteIP == mClientInfo.mLocalIP) { 210c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber // Disallow connections from the local interface 211c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber // for security reasons. 212c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mNetSession->destroySession(sessionID); 213c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber break; 214c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } 215d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 216c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK(msg->findInt32( 217c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber "server-port", &mClientInfo.mLocalPort)); 218c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mPlaybackSessionID = -1; 219c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 220c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientSessionID = sessionID; 221c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 222c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber ALOGI("We now have a client (%d) connected.", sessionID); 223d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 224ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = AWAITING_CLIENT_SETUP; 225ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 226d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = sendM1(sessionID); 227d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK_EQ(err, (status_t)OK); 228d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 229d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 230d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 231d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case ANetworkSession::kWhatData: 232d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 233b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber status_t err = onReceiveClientData(msg); 234b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 235b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (err != OK) { 236ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClient->onDisplayError( 237ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber IRemoteDisplayClient::kDisplayErrorUnknown); 238b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 239d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 240d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 241d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 242d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber default: 243d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber TRESPASS(); 244d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 245d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 246d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 247d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 248d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case kWhatStop: 249d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 250eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber CHECK(msg->senderAwaitsResponse(&mStopReplyID)); 251d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 252ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber CHECK_LT(mState, AWAITING_CLIENT_TEARDOWN); 253ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 254ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber if (mState >= AWAITING_CLIENT_PLAY) { 255ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber // We have a session, i.e. a previous SETUP succeeded. 256ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 257ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber status_t err = sendM5( 258ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber mClientSessionID, true /* requestShutdown */); 2590b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber 260ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber if (err == OK) { 261ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = AWAITING_CLIENT_TEARDOWN; 262ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 263ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber (new AMessage(kWhatTeardownTriggerTimedOut, id()))->post( 264ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber kTeardownTriggerTimeouSecs * 1000000ll); 265ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 266ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber break; 267ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber } 268ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 269ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber // fall through. 270b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 271d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 272eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber finishStop(); 273d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 274d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 275d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 276d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case kWhatReapDeadClients: 277d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 278d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mReaperPending = false; 279d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 280c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClientSessionID == 0 281c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber || mClientInfo.mPlaybackSession == NULL) { 282c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber break; 283d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 284d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 285c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClientInfo.mPlaybackSession->getLastLifesignUs() 286c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber + kPlaybackSessionTimeoutUs < ALooper::GetNowUs()) { 287c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber ALOGI("playback session timed out, reaping."); 288c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 289ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mNetSession->destroySession(mClientSessionID); 290ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClientSessionID = 0; 291ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber 292ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClient->onDisplayError( 293ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber IRemoteDisplayClient::kDisplayErrorUnknown); 294c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } else { 295d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber scheduleReaper(); 296d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 297d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 298d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 299d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 300d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case kWhatPlaybackSessionNotify: 301d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber { 302d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID; 303d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findInt32("playbackSessionID", &playbackSessionID)); 304d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 305d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t what; 306d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findInt32("what", &what)); 307d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 308c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (what == PlaybackSession::kWhatSessionDead) { 309c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber ALOGI("playback session wants to quit."); 310c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 311ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClient->onDisplayError( 312ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber IRemoteDisplayClient::kDisplayErrorUnknown); 313c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } else if (what == PlaybackSession::kWhatSessionEstablished) { 314c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClient != NULL) { 315c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClient->onDisplayConnected( 316c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mPlaybackSession->getSurfaceTexture(), 317c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mPlaybackSession->width(), 318c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mPlaybackSession->height(), 3190328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mUsingHDCP 3200328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber ? IRemoteDisplayClient::kDisplayFlagSecure 3210328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber : 0); 322c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } 323ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 324ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber if (mState == ABOUT_TO_PLAY) { 325ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = PLAYING; 326ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber } 32796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber } else if (what == PlaybackSession::kWhatSessionDestroyed) { 32896fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber disconnectClient2(); 329c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } else { 330c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK_EQ(what, PlaybackSession::kWhatBinaryData); 331d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 332c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber int32_t channel; 333c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK(msg->findInt32("channel", &channel)); 334d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 335c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber sp<ABuffer> data; 336c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK(msg->findBuffer("data", &data)); 337d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 338c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK_LE(channel, 0xffu); 339c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK_LE(data->size(), 0xffffu); 340d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 341c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber int32_t sessionID; 342c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK(msg->findInt32("sessionID", &sessionID)); 343d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 344c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber char header[4]; 345c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber header[0] = '$'; 346c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber header[1] = channel; 347c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber header[2] = data->size() >> 8; 348c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber header[3] = data->size() & 0xff; 349d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 350c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mNetSession->sendRequest( 351c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber sessionID, header, sizeof(header)); 352d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 353c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mNetSession->sendRequest( 354c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber sessionID, data->data(), data->size()); 355d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 356d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 357d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 358d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 359b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber case kWhatKeepAlive: 360b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber { 361b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber int32_t sessionID; 362b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber CHECK(msg->findInt32("sessionID", &sessionID)); 363b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 364c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClientSessionID != sessionID) { 365b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber // Obsolete event, client is already gone. 366b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber break; 367b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber } 368b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 369b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber sendM16(sessionID); 370b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber break; 371b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber } 372b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 373ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber case kWhatTeardownTriggerTimedOut: 374ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber { 375ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber if (mState == AWAITING_CLIENT_TEARDOWN) { 376ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber ALOGI("TEARDOWN trigger timed out, forcing disconnection."); 377ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 378ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber CHECK_NE(mStopReplyID, 0); 379ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber finishStop(); 380ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber break; 381ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber } 382ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber break; 383ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber } 384ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 385b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber case kWhatHDCPNotify: 386b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber { 387b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber int32_t msgCode, ext1, ext2; 388b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber CHECK(msg->findInt32("msg", &msgCode)); 389b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber CHECK(msg->findInt32("ext1", &ext1)); 390b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber CHECK(msg->findInt32("ext2", &ext2)); 391b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 392eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber ALOGI("Saw HDCP notification code %d, ext1 %d, ext2 %d", 393b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber msgCode, ext1, ext2); 394b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 395b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber switch (msgCode) { 396b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber case HDCPModule::HDCP_INITIALIZATION_COMPLETE: 397b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber { 398b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mHDCPInitializationComplete = true; 399b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 400b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (mSetupTriggerDeferred) { 401b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mSetupTriggerDeferred = false; 402b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 403ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber sendM5(mClientSessionID, false /* requestShutdown */); 404b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 405b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber break; 406b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 407b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 408eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber case HDCPModule::HDCP_SHUTDOWN_COMPLETE: 409ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber case HDCPModule::HDCP_SHUTDOWN_FAILED: 410eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber { 411ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber // Ugly hack to make sure that the call to 412ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber // HDCPObserver::notify is completely handled before 413ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber // we clear the HDCP instance and unload the shared 414ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber // library :( 415ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber (new AMessage(kWhatFinishStop2, id()))->post(300000ll); 416eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber break; 417eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber } 418eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber 419b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber default: 420b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber { 421ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber ALOGE("HDCP failure, shutting down."); 422ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber 423ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClient->onDisplayError( 424ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber IRemoteDisplayClient::kDisplayErrorUnknown); 425b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber break; 426b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 427b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 428b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber break; 429b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 430ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber 431ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber case kWhatFinishStop2: 432ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber { 433ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber finishStop2(); 434ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber break; 435ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber } 436b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 437d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber default: 438d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber TRESPASS(); 439d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 440d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 441d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 442d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::registerResponseHandler( 443d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, int32_t cseq, HandleRTSPResponseFunc func) { 444d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ResponseID id; 445d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber id.mSessionID = sessionID; 446d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber id.mCSeq = cseq; 447d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mResponseHandlers.add(id, func); 448d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 449d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 450d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM1(int32_t sessionID) { 451d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString request = "OPTIONS * RTSP/1.0\r\n"; 452d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&request, mNextCSeq); 453d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 454d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append( 455d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "Require: org.wfa.wfd1.0\r\n" 456d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "\r\n"); 457d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 458d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = 459d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mNetSession->sendRequest(sessionID, request.c_str(), request.size()); 460d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 461d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 462d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 463d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 464d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 465d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber registerResponseHandler( 466d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM1Response); 467d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 468d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ++mNextCSeq; 469d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 470d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return OK; 471d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 472d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 473d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM3(int32_t sessionID) { 474d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString body = 475b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber "wfd_content_protection\r\n" 476d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "wfd_video_formats\r\n" 477d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "wfd_audio_codecs\r\n" 478d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "wfd_client_rtp_ports\r\n"; 479d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 480d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString request = "GET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n"; 481d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&request, mNextCSeq); 482d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 483d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append("Content-Type: text/parameters\r\n"); 484d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append(StringPrintf("Content-Length: %d\r\n", body.size())); 485d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append("\r\n"); 486d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append(body); 487d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 488d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = 489d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mNetSession->sendRequest(sessionID, request.c_str(), request.size()); 490d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 491d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 492d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 493d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 494d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 495d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber registerResponseHandler( 496d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM3Response); 497d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 498d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ++mNextCSeq; 499d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 500d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return OK; 501d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 502d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 503d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM4(int32_t sessionID) { 504d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // wfd_video_formats: 505d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 1 byte "native" 506d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 1 byte "preferred-display-mode-supported" 0 or 1 507d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // one or more avc codec structures 508d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 1 byte profile 509d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 1 byte level 510d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 4 byte CEA mask 511d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 4 byte VESA mask 512d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 4 byte HH mask 513d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 1 byte latency 514d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 2 byte min-slice-slice 515d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 2 byte slice-enc-params 516d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // 1 byte framerate-control-support 517d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // max-hres (none or 2 byte) 518d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // max-vres (none or 2 byte) 519d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 520c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK_EQ(sessionID, mClientSessionID); 521d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 522bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber AString transportString = "UDP"; 523bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber 524bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber char val[PROPERTY_VALUE_MAX]; 525bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber if (property_get("media.wfd.enable-tcp", val, NULL) 526bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber && (!strcasecmp("true", val) || !strcmp("1", val))) { 527bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber ALOGI("Using TCP transport."); 528bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber transportString = "TCP"; 529bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } 530bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber 531b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber // For 720p60: 532b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber // use "30 00 02 02 00000040 00000000 00000000 00 0000 0000 00 none none\r\n" 533b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber // For 720p30: 534b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber // use "28 00 02 02 00000020 00000000 00000000 00 0000 0000 00 none none\r\n" 53572ff5903df5f409ea83f74c363a52f0745ced8b8Andreas Huber // For 720p24: 53672ff5903df5f409ea83f74c363a52f0745ced8b8Andreas Huber // use "78 00 02 02 00008000 00000000 00000000 00 0000 0000 00 none none\r\n" 537d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString body = StringPrintf( 538d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "wfd_video_formats: " 5394a8b9a2363b7b7b4f98022e6d9aae8b8aa8e35e5Andreas Huber "28 00 02 02 00000020 00000000 00000000 00 0000 0000 00 none none\r\n" 540e7bd24af08ef0722fb124a550662bcec48c56f86Andreas Huber "wfd_audio_codecs: %s\r\n" 541de799a74064a363d26f4c1bbc5a59d1b7127f49fAndreas Huber "wfd_presentation_URL: rtsp://%s/wfd1.0/streamid=0 none\r\n" 542d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber "wfd_client_rtp_ports: RTP/AVP/%s;unicast %d 0 mode=play\r\n", 543e7bd24af08ef0722fb124a550662bcec48c56f86Andreas Huber (mUsingPCMAudio 544e7bd24af08ef0722fb124a550662bcec48c56f86Andreas Huber ? "LPCM 00000002 00" // 2 ch PCM 48kHz 545e7bd24af08ef0722fb124a550662bcec48c56f86Andreas Huber : "AAC 00000001 00"), // 2 ch AAC 48kHz 546d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber mClientInfo.mLocalIP.c_str(), transportString.c_str(), mChosenRTPPort); 547d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 548d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString request = "SET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n"; 549d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&request, mNextCSeq); 550d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 551d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append("Content-Type: text/parameters\r\n"); 552d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append(StringPrintf("Content-Length: %d\r\n", body.size())); 553d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append("\r\n"); 554d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append(body); 555d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 556d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = 557d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mNetSession->sendRequest(sessionID, request.c_str(), request.size()); 558d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 559d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 560d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 561d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 562d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 563d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber registerResponseHandler( 564d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM4Response); 565d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 566d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ++mNextCSeq; 567d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 568d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return OK; 569d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 570d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 571ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huberstatus_t WifiDisplaySource::sendM5(int32_t sessionID, bool requestShutdown) { 572ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber AString body = "wfd_trigger_method: "; 573ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber if (requestShutdown) { 574ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber ALOGI("Sending TEARDOWN trigger."); 575ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber body.append("TEARDOWN"); 576ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber } else { 577ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber body.append("SETUP"); 578ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber } 579ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber 580ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber body.append("\r\n"); 581d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 582d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString request = "SET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n"; 583d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&request, mNextCSeq); 584d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 585d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append("Content-Type: text/parameters\r\n"); 586d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append(StringPrintf("Content-Length: %d\r\n", body.size())); 587d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append("\r\n"); 588d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber request.append(body); 589d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 590d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = 591d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mNetSession->sendRequest(sessionID, request.c_str(), request.size()); 592d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 593d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 594d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return err; 595d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 596d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 597d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber registerResponseHandler( 598d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM5Response); 599d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 600d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ++mNextCSeq; 601d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 602d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return OK; 603d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 604d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 605b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huberstatus_t WifiDisplaySource::sendM16(int32_t sessionID) { 606b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber AString request = "GET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n"; 607b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber AppendCommonResponse(&request, mNextCSeq); 608b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 609c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK_EQ(sessionID, mClientSessionID); 610c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber request.append( 611c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber StringPrintf("Session: %d\r\n", mClientInfo.mPlaybackSessionID)); 612a438123bd96c7faf145683876702387efe5628d9Andreas Huber request.append("\r\n"); // Empty body 613b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 614b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber status_t err = 615b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber mNetSession->sendRequest(sessionID, request.c_str(), request.size()); 616b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 617b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber if (err != OK) { 618b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber return err; 619b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber } 620b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 621b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber registerResponseHandler( 622b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM16Response); 623b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 624b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber ++mNextCSeq; 625b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 626b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber return OK; 627b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber} 628b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 629d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM1Response( 630d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, const sp<ParsedMessage> &msg) { 631d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t statusCode; 632d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!msg->getStatusCode(&statusCode)) { 633d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_MALFORMED; 634d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 635d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 636d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (statusCode != 200) { 637d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_UNSUPPORTED; 638d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 639d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 640d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return OK; 641d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 642d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 643d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber// sink_audio_list := ("LPCM"|"AAC"|"AC3" HEXDIGIT*8 HEXDIGIT*2) 644d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber// (", " sink_audio_list)* 645d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huberstatic void GetAudioModes(const char *s, const char *prefix, uint32_t *modes) { 646d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber *modes = 0; 647d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 648d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber size_t prefixLen = strlen(prefix); 649d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 650d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber while (*s != '0') { 651d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (!strncmp(s, prefix, prefixLen) && s[prefixLen] == ' ') { 652d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber unsigned latency; 653d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (sscanf(&s[prefixLen + 1], "%08x %02x", modes, &latency) != 2) { 654d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber *modes = 0; 655d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 656d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 657d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber return; 658d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 659d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 660d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber char *commaPos = strchr(s, ','); 661d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (commaPos != NULL) { 662d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber s = commaPos + 1; 663d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 664d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber while (isspace(*s)) { 665d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ++s; 666d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 667d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } else { 668d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber break; 669d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 670d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 671d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber} 672d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 673d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM3Response( 674d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, const sp<ParsedMessage> &msg) { 675d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t statusCode; 676d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!msg->getStatusCode(&statusCode)) { 677d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_MALFORMED; 678d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 679d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 680d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (statusCode != 200) { 681d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_UNSUPPORTED; 682d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 683d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 684b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sp<Parameters> params = 685b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber Parameters::Parse(msg->getContent(), strlen(msg->getContent())); 686b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 687b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (params == NULL) { 688b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 689b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 690b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 691b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber AString value; 692d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (!params->findParameter("wfd_client_rtp_ports", &value)) { 693d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGE("Sink doesn't report its choice of wfd_client_rtp_ports."); 694d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber return ERROR_MALFORMED; 695d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 696d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 697d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber unsigned port0, port1; 698d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (sscanf(value.c_str(), 699d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber "RTP/AVP/UDP;unicast %u %u mode=play", 700d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber &port0, 701d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber &port1) != 2 702d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber || port0 == 0 || port0 > 65535 || port1 != 0) { 703d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGE("Sink chose its wfd_client_rtp_ports poorly (%s)", 704d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber value.c_str()); 705d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 706d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber return ERROR_MALFORMED; 707d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 708d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 709d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber mChosenRTPPort = port0; 710d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 711d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (!params->findParameter("wfd_audio_codecs", &value)) { 712d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGE("Sink doesn't report its choice of wfd_audio_codecs."); 713d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber return ERROR_MALFORMED; 714d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 715d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 716d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (value == "none") { 717d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGE("Sink doesn't support audio at all."); 718d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber return ERROR_UNSUPPORTED; 719d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 720d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 721d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber uint32_t modes; 722d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber GetAudioModes(value.c_str(), "AAC", &modes); 723d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 724d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber bool supportsAAC = (modes & 1) != 0; // AAC 2ch 48kHz 725d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 726d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber GetAudioModes(value.c_str(), "LPCM", &modes); 727d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 728d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber bool supportsPCM = (modes & 2) != 0; // LPCM 2ch 48kHz 729d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 730d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber char val[PROPERTY_VALUE_MAX]; 731d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber if (supportsPCM 732d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber && property_get("media.wfd.use-pcm-audio", val, NULL) 733d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber && (!strcasecmp("true", val) || !strcmp("1", val))) { 734d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGI("Using PCM audio."); 735d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber mUsingPCMAudio = true; 736d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } else if (supportsAAC) { 737d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGI("Using AAC audio."); 738d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber mUsingPCMAudio = false; 739d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } else if (supportsPCM) { 740d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGI("Using PCM audio."); 741d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber mUsingPCMAudio = true; 742d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } else { 743d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber ALOGI("Sink doesn't support an audio format we do."); 744d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber return ERROR_UNSUPPORTED; 745d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber } 746d243c04534d1b74bd66625c5c96a9b918d8838bfAndreas Huber 7470328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mUsingHDCP = false; 748b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (!params->findParameter("wfd_content_protection", &value)) { 7490328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber ALOGI("Sink doesn't appear to support content protection."); 7500328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber } else if (value == "none") { 7510328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber ALOGI("Sink does not support content protection."); 7520328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber } else { 7530328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mUsingHDCP = true; 754b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 7550328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber bool isHDCP2_0 = false; 7560328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber if (value.startsWith("HDCP2.0 ")) { 7570328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber isHDCP2_0 = true; 7580328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber } else if (!value.startsWith("HDCP2.1 ")) { 7590328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber ALOGE("malformed wfd_content_protection: '%s'", value.c_str()); 760b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 7610328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber return ERROR_MALFORMED; 7620328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber } 763b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 7640328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber int32_t hdcpPort; 7650328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber if (!ParsedMessage::GetInt32Attribute( 7660328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber value.c_str() + 8, "port", &hdcpPort) 7670328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber || hdcpPort < 1 || hdcpPort > 65535) { 7680328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber return ERROR_MALFORMED; 7690328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber } 770b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 7710328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mIsHDCP2_0 = isHDCP2_0; 7720328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mHDCPPort = hdcpPort; 773b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 7740328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber status_t err = makeHDCP(); 7750328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber if (err != OK) { 7760328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber ALOGE("Unable to instantiate HDCP component."); 7770328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber return err; 7780328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber } 779b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 780b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 781d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return sendM4(sessionID); 782d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 783d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 784d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM4Response( 785d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, const sp<ParsedMessage> &msg) { 786d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t statusCode; 787d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!msg->getStatusCode(&statusCode)) { 788d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_MALFORMED; 789d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 790d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 791d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (statusCode != 200) { 792d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_UNSUPPORTED; 793d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 794d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 7950328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber if (mUsingHDCP && !mHDCPInitializationComplete) { 796b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber ALOGI("Deferring SETUP trigger until HDCP initialization completes."); 797b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 798b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mSetupTriggerDeferred = true; 799b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return OK; 800b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 801b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 802ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber return sendM5(sessionID, false /* requestShutdown */); 803d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 804d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 805d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM5Response( 806d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, const sp<ParsedMessage> &msg) { 807d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t statusCode; 808d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!msg->getStatusCode(&statusCode)) { 809d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_MALFORMED; 810d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 811d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 812d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (statusCode != 200) { 813d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return ERROR_UNSUPPORTED; 814d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 815d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 816d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return OK; 817d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 818d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 819b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huberstatus_t WifiDisplaySource::onReceiveM16Response( 820b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber int32_t sessionID, const sp<ParsedMessage> &msg) { 821b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber // If only the response was required to include a "Session:" header... 822b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 823c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK_EQ(sessionID, mClientSessionID); 824b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 825c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClientInfo.mPlaybackSession != NULL) { 826c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mPlaybackSession->updateLiveness(); 827b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 828b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber scheduleKeepAlive(sessionID); 829b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber } 830b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 831b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber return OK; 832b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber} 833b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 834d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::scheduleReaper() { 835d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (mReaperPending) { 836d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return; 837d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 838d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 839d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mReaperPending = true; 840d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber (new AMessage(kWhatReapDeadClients, id()))->post(kReaperIntervalUs); 841d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 842d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 843b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Hubervoid WifiDisplaySource::scheduleKeepAlive(int32_t sessionID) { 844b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber // We need to send updates at least 5 secs before the timeout is set to 845b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber // expire, make sure the timeout is greater than 5 secs to begin with. 846b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber CHECK_GT(kPlaybackSessionTimeoutUs, 5000000ll); 847b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 848b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber sp<AMessage> msg = new AMessage(kWhatKeepAlive, id()); 849b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber msg->setInt32("sessionID", sessionID); 850b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber msg->post(kPlaybackSessionTimeoutUs - 5000000ll); 851b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber} 852b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 853b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onReceiveClientData(const sp<AMessage> &msg) { 854d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID; 855d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findInt32("sessionID", &sessionID)); 856d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 857d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<RefBase> obj; 858d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK(msg->findObject("data", &obj)); 859d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 860d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<ParsedMessage> data = 861d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber static_cast<ParsedMessage *>(obj.get()); 862d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 863d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ALOGV("session %d received '%s'", 864d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sessionID, data->debugString().c_str()); 865d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 866d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString method; 867d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString uri; 868d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber data->getRequestField(0, &method); 869d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 870d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq; 871d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!data->findInt32("cseq", &cseq)) { 872d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "400 Bad Request", -1 /* cseq */); 873b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 874d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 875d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 876d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (method.startsWith("RTSP/")) { 877d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // This is a response. 878d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 879d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ResponseID id; 880d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber id.mSessionID = sessionID; 881d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber id.mCSeq = cseq; 882d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 883d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ssize_t index = mResponseHandlers.indexOfKey(id); 884d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 885d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (index < 0) { 886d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ALOGW("Received unsolicited server response, cseq %d", cseq); 887b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 888d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 889d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 890d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber HandleRTSPResponseFunc func = mResponseHandlers.valueAt(index); 891d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber mResponseHandlers.removeItemsAt(index); 892d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 893d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = (this->*func)(sessionID, data); 894d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 895d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 896d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber ALOGW("Response handler for session %d, cseq %d returned " 897d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "err %d (%s)", 898d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sessionID, cseq, err, strerror(-err)); 899d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 900b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 901d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 902d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 903b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return OK; 904b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 905d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 906b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber AString version; 907b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber data->getRequestField(2, &version); 908b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (!(version == AString("RTSP/1.0"))) { 909b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sendErrorResponse(sessionID, "505 RTSP Version not supported", cseq); 910b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_UNSUPPORTED; 911b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 912d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 913b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber status_t err; 914b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (method == "OPTIONS") { 915b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = onOptionsRequest(sessionID, cseq, data); 916b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } else if (method == "SETUP") { 917b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = onSetupRequest(sessionID, cseq, data); 918b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } else if (method == "PLAY") { 919b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = onPlayRequest(sessionID, cseq, data); 920b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } else if (method == "PAUSE") { 921b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = onPauseRequest(sessionID, cseq, data); 922b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } else if (method == "TEARDOWN") { 923b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = onTeardownRequest(sessionID, cseq, data); 924b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } else if (method == "GET_PARAMETER") { 925b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = onGetParameterRequest(sessionID, cseq, data); 926b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } else if (method == "SET_PARAMETER") { 927b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = onSetParameterRequest(sessionID, cseq, data); 928b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } else { 929b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sendErrorResponse(sessionID, "405 Method Not Allowed", cseq); 930d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 931b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = ERROR_UNSUPPORTED; 932b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 933d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 934b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 935d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 936d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 937b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onOptionsRequest( 938d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 939d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq, 940d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data) { 941d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID; 942d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<PlaybackSession> playbackSession = 943d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber findPlaybackSession(data, &playbackSessionID); 944d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 945d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (playbackSession != NULL) { 946d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber playbackSession->updateLiveness(); 947d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 948d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 949d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response = "RTSP/1.0 200 OK\r\n"; 950d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq); 951d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 952d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append( 953b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber "Public: org.wfa.wfd1.0, SETUP, TEARDOWN, PLAY, PAUSE, " 954d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "GET_PARAMETER, SET_PARAMETER\r\n"); 955d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 956d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 957d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 958d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = mNetSession->sendRequest(sessionID, response.c_str()); 959d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 960b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (err == OK) { 961b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = sendM3(sessionID); 962b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 963b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 964b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 965d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 966d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 967b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onSetupRequest( 968d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 969d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq, 970d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data) { 971c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber CHECK_EQ(sessionID, mClientSessionID); 972c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (mClientInfo.mPlaybackSessionID != -1) { 973b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber // We only support a single playback session per client. 974b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber // This is due to the reversed keep-alive design in the wfd specs... 975b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber sendErrorResponse(sessionID, "400 Bad Request", cseq); 976b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 977b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber } 978b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 979d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString transport; 980d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!data->findString("transport", &transport)) { 981d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "400 Bad Request", cseq); 982b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 983d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 984d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 98590a92053219ae50ddf4bb54e3d54db2d309e2b8dAndreas Huber Sender::TransportMode transportMode = Sender::TRANSPORT_UDP; 986d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 987d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int clientRtp, clientRtcp; 988d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (transport.startsWith("RTP/AVP/TCP;")) { 989d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString interleaved; 990bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber if (ParsedMessage::GetAttribute( 991d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber transport.c_str(), "interleaved", &interleaved) 992bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber && sscanf(interleaved.c_str(), "%d-%d", 993bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber &clientRtp, &clientRtcp) == 2) { 99490a92053219ae50ddf4bb54e3d54db2d309e2b8dAndreas Huber transportMode = Sender::TRANSPORT_TCP_INTERLEAVED; 995bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } else { 996bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber bool badRequest = false; 997bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber 998bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber AString clientPort; 999bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber if (!ParsedMessage::GetAttribute( 1000bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber transport.c_str(), "client_port", &clientPort)) { 1001bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber badRequest = true; 1002bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } else if (sscanf(clientPort.c_str(), "%d-%d", 1003bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber &clientRtp, &clientRtcp) == 2) { 1004bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } else if (sscanf(clientPort.c_str(), "%d", &clientRtp) == 1) { 1005bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber // No RTCP. 1006bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber clientRtcp = -1; 1007bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } else { 1008bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber badRequest = true; 1009bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } 1010bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber 1011bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber if (badRequest) { 1012bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber sendErrorResponse(sessionID, "400 Bad Request", cseq); 1013b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1014bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } 1015d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 101690a92053219ae50ddf4bb54e3d54db2d309e2b8dAndreas Huber transportMode = Sender::TRANSPORT_TCP; 1017bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } 1018d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else if (transport.startsWith("RTP/AVP;unicast;") 1019d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber || transport.startsWith("RTP/AVP/UDP;unicast;")) { 1020d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber bool badRequest = false; 1021d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1022d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString clientPort; 1023d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!ParsedMessage::GetAttribute( 1024d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber transport.c_str(), "client_port", &clientPort)) { 1025d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber badRequest = true; 1026d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else if (sscanf(clientPort.c_str(), "%d-%d", 1027d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber &clientRtp, &clientRtcp) == 2) { 1028d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else if (sscanf(clientPort.c_str(), "%d", &clientRtp) == 1) { 1029d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber // No RTCP. 1030d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber clientRtcp = -1; 1031d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else { 1032d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber badRequest = true; 1033d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1034d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1035d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (badRequest) { 1036d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "400 Bad Request", cseq); 1037b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1038d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1039d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#if 1 1040a438123bd96c7faf145683876702387efe5628d9Andreas Huber // The older LG dongles doesn't specify client_port=xxx apparently. 1041d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else if (transport == "RTP/AVP/UDP;unicast") { 1042d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber clientRtp = 19000; 104328e17ed7e2fbb254fb99481b74db85e427c905eeAndreas Huber clientRtcp = -1; 1044d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#endif 1045d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else { 1046d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "461 Unsupported Transport", cseq); 1047b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_UNSUPPORTED; 1048d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1049d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1050d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID = makeUniquePlaybackSessionID(); 1051d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1052d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<AMessage> notify = new AMessage(kWhatPlaybackSessionNotify, id()); 1053d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber notify->setInt32("playbackSessionID", playbackSessionID); 1054d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber notify->setInt32("sessionID", sessionID); 1055d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1056d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<PlaybackSession> playbackSession = 10570b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber new PlaybackSession( 10580328ec08dc1e90caa2a9e0c4e107d8ddaa74af20Andreas Huber mNetSession, notify, mInterfaceAddr, mHDCP); 1059d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1060d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber looper()->registerHandler(playbackSession); 1061d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1062d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString uri; 1063d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber data->getRequestField(1, &uri); 1064d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1065d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (strncasecmp("rtsp://", uri.c_str(), 7)) { 1066d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "400 Bad Request", cseq); 1067b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1068d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1069d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1070d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!(uri.startsWith("rtsp://") && uri.endsWith("/wfd1.0/streamid=0"))) { 1071d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "404 Not found", cseq); 1072b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1073d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1074d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1075d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = playbackSession->init( 1076c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mRemoteIP.c_str(), 1077d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber clientRtp, 1078d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber clientRtcp, 1079e7bd24af08ef0722fb124a550662bcec48c56f86Andreas Huber transportMode, 1080e7bd24af08ef0722fb124a550662bcec48c56f86Andreas Huber mUsingPCMAudio); 1081d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1082d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (err != OK) { 1083d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber looper()->unregisterHandler(playbackSession->id()); 1084d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber playbackSession.clear(); 1085d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1086d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1087d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber switch (err) { 1088d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case OK: 1089d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber break; 1090d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber case -ENOENT: 1091d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "404 Not Found", cseq); 1092b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1093d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber default: 1094d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "403 Forbidden", cseq); 1095b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1096d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1097d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1098c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mPlaybackSessionID = playbackSessionID; 1099c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientInfo.mPlaybackSession = playbackSession; 1100b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber 1101d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response = "RTSP/1.0 200 OK\r\n"; 1102d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq, playbackSessionID); 1103d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 110490a92053219ae50ddf4bb54e3d54db2d309e2b8dAndreas Huber if (transportMode == Sender::TRANSPORT_TCP_INTERLEAVED) { 1105d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append( 1106d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber StringPrintf( 1107d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "Transport: RTP/AVP/TCP;interleaved=%d-%d;", 1108d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber clientRtp, clientRtcp)); 1109d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else { 1110d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t serverRtp = playbackSession->getRTPPort(); 1111d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1112bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber AString transportString = "UDP"; 111390a92053219ae50ddf4bb54e3d54db2d309e2b8dAndreas Huber if (transportMode == Sender::TRANSPORT_TCP) { 1114bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber transportString = "TCP"; 1115bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber } 1116bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber 1117d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (clientRtcp >= 0) { 1118d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append( 1119d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber StringPrintf( 1120bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber "Transport: RTP/AVP/%s;unicast;client_port=%d-%d;" 1121d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "server_port=%d-%d\r\n", 1122bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber transportString.c_str(), 1123d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber clientRtp, clientRtcp, serverRtp, serverRtp + 1)); 1124d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } else { 1125d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append( 1126d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber StringPrintf( 1127bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber "Transport: RTP/AVP/%s;unicast;client_port=%d;" 1128d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "server_port=%d\r\n", 1129bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber transportString.c_str(), 1130d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber clientRtp, serverRtp)); 1131d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1132d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1133d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1134d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1135d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1136d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber err = mNetSession->sendRequest(sessionID, response.c_str()); 1137b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1138b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (err != OK) { 1139b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1140b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 1141d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1142ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = AWAITING_CLIENT_PLAY; 1143ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 1144d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber scheduleReaper(); 1145b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber scheduleKeepAlive(sessionID); 1146b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1147b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return OK; 1148d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1149d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1150b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onPlayRequest( 1151d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 1152d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq, 1153d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data) { 1154d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID; 1155d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<PlaybackSession> playbackSession = 1156d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber findPlaybackSession(data, &playbackSessionID); 1157d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1158d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (playbackSession == NULL) { 1159d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "454 Session Not Found", cseq); 1160b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1161d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1162d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1163ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber ALOGI("Received PLAY request."); 1164ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber 1165d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = playbackSession->play(); 1166d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK_EQ(err, (status_t)OK); 1167d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1168d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response = "RTSP/1.0 200 OK\r\n"; 1169d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq, playbackSessionID); 1170d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("Range: npt=now-\r\n"); 1171d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1172d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1173d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber err = mNetSession->sendRequest(sessionID, response.c_str()); 1174b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1175b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (err != OK) { 1176b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1177b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 11780b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber 1179bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber playbackSession->finishPlay(); 1180b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1181ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber CHECK_EQ(mState, AWAITING_CLIENT_PLAY); 1182ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = ABOUT_TO_PLAY; 1183ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 1184b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return OK; 1185d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1186d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1187b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onPauseRequest( 1188d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 1189d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq, 1190d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data) { 1191d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID; 1192d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<PlaybackSession> playbackSession = 1193d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber findPlaybackSession(data, &playbackSessionID); 1194d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1195d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (playbackSession == NULL) { 1196d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "454 Session Not Found", cseq); 1197b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1198d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1199d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1200d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = playbackSession->pause(); 1201d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber CHECK_EQ(err, (status_t)OK); 1202d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1203d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response = "RTSP/1.0 200 OK\r\n"; 1204d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq, playbackSessionID); 1205d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1206d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1207d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber err = mNetSession->sendRequest(sessionID, response.c_str()); 1208b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1209b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1210d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1211d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1212b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onTeardownRequest( 1213d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 1214d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq, 1215d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data) { 1216ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber ALOGI("Received TEARDOWN request."); 1217ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber 1218d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID; 1219d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<PlaybackSession> playbackSession = 1220d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber findPlaybackSession(data, &playbackSessionID); 1221d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1222d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (playbackSession == NULL) { 1223d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "454 Session Not Found", cseq); 1224b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1225d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1226d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1227d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response = "RTSP/1.0 200 OK\r\n"; 1228d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq, playbackSessionID); 1229c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber response.append("Connection: close\r\n"); 1230d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1231d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1232ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mNetSession->sendRequest(sessionID, response.c_str()); 1233c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 1234ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber if (mState == AWAITING_CLIENT_TEARDOWN) { 1235ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber CHECK_NE(mStopReplyID, 0); 1236eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber finishStop(); 1237ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber } else { 1238ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClient->onDisplayError(IRemoteDisplayClient::kDisplayErrorUnknown); 1239ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber } 1240b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1241b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return OK; 1242d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1243d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1244eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Hubervoid WifiDisplaySource::finishStop() { 1245ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber ALOGV("finishStop"); 1246ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber 1247ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = STOPPING; 1248ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber 124996fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber disconnectClientAsync(); 125096fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber} 125196fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber 125296fc6cc65ca93009a759a3a874b82a35771b9714Andreas Hubervoid WifiDisplaySource::finishStopAfterDisconnectingClient() { 125396fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber ALOGV("finishStopAfterDisconnectingClient"); 125496fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber 1255ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber if (mHDCP != NULL) { 1256ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber ALOGI("Initiating HDCP shutdown."); 1257ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber mHDCP->shutdownAsync(); 1258eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber return; 1259ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber } 1260ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber 1261eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber finishStop2(); 1262eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber} 1263eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber 1264eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Hubervoid WifiDisplaySource::finishStop2() { 1265ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber ALOGV("finishStop2"); 1266ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber 1267bbe96f0f05a1f1a1b3cfec0d124cb0d63c1ebf2aAndreas Huber if (mHDCP != NULL) { 1268bbe96f0f05a1f1a1b3cfec0d124cb0d63c1ebf2aAndreas Huber mHDCP->setObserver(NULL); 1269bbe96f0f05a1f1a1b3cfec0d124cb0d63c1ebf2aAndreas Huber mHDCPObserver.clear(); 1270bbe96f0f05a1f1a1b3cfec0d124cb0d63c1ebf2aAndreas Huber mHDCP.clear(); 1271bbe96f0f05a1f1a1b3cfec0d124cb0d63c1ebf2aAndreas Huber } 1272eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber 1273ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber if (mSessionID != 0) { 1274ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mNetSession->destroySession(mSessionID); 1275ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mSessionID = 0; 1276ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber } 1277ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber 127896fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber ALOGI("We're stopped."); 1279ad0d97c7cf620e96a0b088dd9461645a3f8900b7Andreas Huber mState = STOPPED; 1280ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber 1281ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber status_t err = OK; 1282ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber 1283ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber sp<AMessage> response = new AMessage; 1284ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber response->setInt32("err", err); 1285eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber response->postReply(mStopReplyID); 1286ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber} 1287ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber 1288b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onGetParameterRequest( 1289d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 1290d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq, 1291d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data) { 1292d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID; 1293d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<PlaybackSession> playbackSession = 1294d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber findPlaybackSession(data, &playbackSessionID); 1295d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1296d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (playbackSession == NULL) { 1297d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "454 Session Not Found", cseq); 1298b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1299d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1300d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1301d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber playbackSession->updateLiveness(); 1302d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1303d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response = "RTSP/1.0 200 OK\r\n"; 1304d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq, playbackSessionID); 1305d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1306d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1307d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = mNetSession->sendRequest(sessionID, response.c_str()); 1308b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1309d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1310d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1311b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onSetParameterRequest( 1312d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 1313d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq, 1314d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data) { 1315d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t playbackSessionID; 1316d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sp<PlaybackSession> playbackSession = 1317d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber findPlaybackSession(data, &playbackSessionID); 1318d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1319d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (playbackSession == NULL) { 1320d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber sendErrorResponse(sessionID, "454 Session Not Found", cseq); 1321b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_MALFORMED; 1322d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1323d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1324b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (strstr(data->getContent(), "wfd_idr_request\r\n")) { 1325b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber playbackSession->requestIDRFrame(); 1326b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 1327496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber 1328d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber playbackSession->updateLiveness(); 1329d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1330d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response = "RTSP/1.0 200 OK\r\n"; 1331d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq, playbackSessionID); 1332d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1333d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1334d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber status_t err = mNetSession->sendRequest(sessionID, response.c_str()); 1335b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1336d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1337d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1338d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber// static 1339d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::AppendCommonResponse( 1340d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString *response, int32_t cseq, int32_t playbackSessionID) { 1341d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber time_t now = time(NULL); 1342d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber struct tm *now2 = gmtime(&now); 1343d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber char buf[128]; 1344d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber strftime(buf, sizeof(buf), "%a, %d %b %Y %H:%M:%S %z", now2); 1345d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1346d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->append("Date: "); 1347d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->append(buf); 1348d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->append("\r\n"); 1349d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1350d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->append("Server: Mine/1.0\r\n"); 1351d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1352d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (cseq >= 0) { 1353d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->append(StringPrintf("CSeq: %d\r\n", cseq)); 1354d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1355d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1356d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (playbackSessionID >= 0ll) { 1357d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response->append( 1358d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber StringPrintf( 1359d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber "Session: %d;timeout=%lld\r\n", 1360d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber playbackSessionID, kPlaybackSessionTimeoutSecs)); 1361d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1362d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1363d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1364d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::sendErrorResponse( 1365d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t sessionID, 1366d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const char *errorDetail, 1367d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber int32_t cseq) { 1368d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AString response; 1369d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("RTSP/1.0 "); 1370d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append(errorDetail); 1371d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1372d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1373d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber AppendCommonResponse(&response, cseq); 1374d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1375d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber response.append("\r\n"); 1376d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1377eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber mNetSession->sendRequest(sessionID, response.c_str()); 1378d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1379d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1380d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberint32_t WifiDisplaySource::makeUniquePlaybackSessionID() const { 1381c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber return rand(); 1382d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1383d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1384d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubersp<WifiDisplaySource::PlaybackSession> WifiDisplaySource::findPlaybackSession( 1385d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber const sp<ParsedMessage> &data, int32_t *playbackSessionID) const { 1386d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber if (!data->findInt32("session", playbackSessionID)) { 1387c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber // XXX the older dongles do not always include a "Session:" header. 1388c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber *playbackSessionID = mClientInfo.mPlaybackSessionID; 1389c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber return mClientInfo.mPlaybackSession; 1390d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1391d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1392c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber if (*playbackSessionID != mClientInfo.mPlaybackSessionID) { 1393d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber return NULL; 1394d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber } 1395d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1396c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber return mClientInfo.mPlaybackSession; 1397c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber} 1398c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 139996fc6cc65ca93009a759a3a874b82a35771b9714Andreas Hubervoid WifiDisplaySource::disconnectClientAsync() { 140096fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber ALOGV("disconnectClient"); 140196fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber 140296fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber if (mClientInfo.mPlaybackSession == NULL) { 140396fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber disconnectClient2(); 140496fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber return; 140596fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber } 140696fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber 140796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber if (mClientInfo.mPlaybackSession != NULL) { 140896fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber ALOGV("Destroying PlaybackSession"); 140996fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber mClientInfo.mPlaybackSession->destroyAsync(); 141096fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber } 141196fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber} 141296fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber 141396fc6cc65ca93009a759a3a874b82a35771b9714Andreas Hubervoid WifiDisplaySource::disconnectClient2() { 141496fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber ALOGV("disconnectClient2"); 141596fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber 1416ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber if (mClientInfo.mPlaybackSession != NULL) { 141796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber looper()->unregisterHandler(mClientInfo.mPlaybackSession->id()); 1418ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClientInfo.mPlaybackSession.clear(); 1419ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber } 1420c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 1421ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber if (mClientSessionID != 0) { 1422c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mNetSession->destroySession(mClientSessionID); 1423c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber mClientSessionID = 0; 1424c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber } 1425c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber 1426ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber mClient->onDisplayDisconnected(); 142796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber 142896fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber finishStopAfterDisconnectingClient(); 1429d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} 1430d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1431b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstruct WifiDisplaySource::HDCPObserver : public BnHDCPObserver { 1432b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber HDCPObserver(const sp<AMessage> ¬ify); 1433b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1434b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber virtual void notify( 1435b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber int msg, int ext1, int ext2, const Parcel *obj); 1436b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1437b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberprivate: 1438b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sp<AMessage> mNotify; 1439b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1440b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber DISALLOW_EVIL_CONSTRUCTORS(HDCPObserver); 1441b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber}; 1442b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1443b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas HuberWifiDisplaySource::HDCPObserver::HDCPObserver( 1444b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber const sp<AMessage> ¬ify) 1445b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber : mNotify(notify) { 1446b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber} 1447b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1448b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Hubervoid WifiDisplaySource::HDCPObserver::notify( 1449b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber int msg, int ext1, int ext2, const Parcel *obj) { 1450b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sp<AMessage> notify = mNotify->dup(); 1451b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber notify->setInt32("msg", msg); 1452b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber notify->setInt32("ext1", ext1); 1453b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber notify->setInt32("ext2", ext2); 1454b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber notify->post(); 1455b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber} 1456b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1457b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::makeHDCP() { 1458b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sp<IServiceManager> sm = defaultServiceManager(); 1459b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sp<IBinder> binder = sm->getService(String16("media.player")); 1460b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sp<IMediaPlayerService> service = interface_cast<IMediaPlayerService>(binder); 1461b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber CHECK(service != NULL); 1462b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1463b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mHDCP = service->makeHDCP(); 1464b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1465b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (mHDCP == NULL) { 1466b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return ERROR_UNSUPPORTED; 1467b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 1468b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1469b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber sp<AMessage> notify = new AMessage(kWhatHDCPNotify, id()); 1470b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mHDCPObserver = new HDCPObserver(notify); 1471b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1472b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber status_t err = mHDCP->setObserver(mHDCPObserver); 1473b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1474b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (err != OK) { 1475b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber ALOGE("Failed to set HDCP observer."); 1476b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1477b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mHDCPObserver.clear(); 1478b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mHDCP.clear(); 1479b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1480b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1481b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 1482b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1483ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber ALOGI("Initiating HDCP negotiation w/ host %s:%d", 1484b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber mClientInfo.mRemoteIP.c_str(), mHDCPPort); 1485b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1486b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber err = mHDCP->initAsync(mClientInfo.mRemoteIP.c_str(), mHDCPPort); 1487b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1488b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber if (err != OK) { 1489b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return err; 1490b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber } 1491b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1492b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber return OK; 1493b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber} 1494b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber 1495d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber} // namespace android 1496d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber 1497