WifiDisplaySource.cpp revision 96fc6cc65ca93009a759a3a874b82a35771b9714
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"
25d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
26b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#include <binder/IServiceManager.h>
270b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber#include <gui/ISurfaceTexture.h>
28b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#include <media/IHDCP.h>
29b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#include <media/IMediaPlayerService.h>
300b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber#include <media/IRemoteDisplayClient.h>
31d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/ABuffer.h>
32d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/ADebug.h>
33d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/AMessage.h>
34d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/MediaErrors.h>
35d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
36bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber#include <arpa/inet.h>
37bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber#include <cutils/properties.h>
38bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
39d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubernamespace android {
40d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
410b73d4730202fcad53aefc4314a06e7b95f442f0Andreas HuberWifiDisplaySource::WifiDisplaySource(
420b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber        const sp<ANetworkSession> &netSession,
430b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber        const sp<IRemoteDisplayClient> &client)
44d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    : mNetSession(netSession),
450b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber      mClient(client),
46d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber      mSessionID(0),
47ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber      mStopReplyID(0),
48c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber      mClientSessionID(0),
49d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber      mReaperPending(false),
50b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber      mNextCSeq(1)
51b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#if REQUIRE_HDCP
52b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber      ,mIsHDCP2_0(false)
53b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber      ,mHDCPPort(0)
54b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber      ,mHDCPInitializationComplete(false)
55b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber      ,mSetupTriggerDeferred(false)
56b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#endif
57b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber{
58d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
59d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
60d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas HuberWifiDisplaySource::~WifiDisplaySource() {
61d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
62d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
63bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huberstatus_t WifiDisplaySource::start(const char *iface) {
64d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> msg = new AMessage(kWhatStart, id());
65bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber    msg->setString("iface", iface);
66d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
67d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> response;
68d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = msg->postAndAwaitResponse(&response);
69d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
70d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
71d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
72d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
73d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
74d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!response->findInt32("err", &err)) {
75d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        err = OK;
76d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
77d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
78d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return err;
79d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
80d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
81d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::stop() {
82d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> msg = new AMessage(kWhatStop, id());
83d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
84d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> response;
85d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = msg->postAndAwaitResponse(&response);
86d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
87d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
88d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
89d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
90d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
91d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!response->findInt32("err", &err)) {
92d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        err = OK;
93d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
94d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
95d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return err;
96d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
97d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
98d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onMessageReceived(const sp<AMessage> &msg) {
99d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    switch (msg->what()) {
100d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatStart:
101d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
102d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            uint32_t replyID;
103d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->senderAwaitsResponse(&replyID));
104d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
105bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            AString iface;
106bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            CHECK(msg->findString("iface", &iface));
107bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
108bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            status_t err = OK;
109bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
110bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            ssize_t colonPos = iface.find(":");
111bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
112bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            unsigned long port;
113bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
114bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            if (colonPos >= 0) {
115bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                const char *s = iface.c_str() + colonPos + 1;
116bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
117bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                char *end;
118bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                port = strtoul(s, &end, 10);
119bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
120bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                if (end == s || *end != '\0' || port > 65535) {
121bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    err = -EINVAL;
122bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                } else {
123bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    iface.erase(colonPos, iface.size() - colonPos);
124bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                }
125bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            } else {
126bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                port = kWifiDisplayDefaultPort;
127bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            }
128bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
129bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            if (err == OK) {
130bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                if (inet_aton(iface.c_str(), &mInterfaceAddr) != 0) {
131bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    sp<AMessage> notify = new AMessage(kWhatRTSPNotify, id());
132d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
133bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    err = mNetSession->createRTSPServer(
134bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                            mInterfaceAddr, port, notify, &mSessionID);
135bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                } else {
136bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    err = -EINVAL;
137bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                }
138bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            }
139d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
140d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sp<AMessage> response = new AMessage;
141d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response->setInt32("err", err);
142d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response->postReply(replyID);
143d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
144d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
145d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
146d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatRTSPNotify:
147d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
148d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            int32_t reason;
149d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->findInt32("reason", &reason));
150d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
151d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            switch (reason) {
152d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                case ANetworkSession::kWhatError:
153d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                {
154d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t sessionID;
155d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("sessionID", &sessionID));
156d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
157d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t err;
158d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("err", &err));
159d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
160d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    AString detail;
161d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findString("detail", &detail));
162d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
163d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    ALOGE("An error occurred in session %d (%d, '%s/%s').",
164d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          sessionID,
165d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          err,
166d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          detail.c_str(),
167d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          strerror(-err));
168d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
169d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mNetSession->destroySession(sessionID);
170d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
171c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    if (sessionID == mClientSessionID) {
172ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                        mClientSessionID = 0;
173c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
174ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                        mClient->onDisplayError(
175ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                                IRemoteDisplayClient::kDisplayErrorUnknown);
176c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    }
177d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    break;
178d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
179d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
180d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                case ANetworkSession::kWhatClientConnected:
181d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                {
182d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t sessionID;
183d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("sessionID", &sessionID));
184d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
185c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    if (mClientSessionID > 0) {
186c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        ALOGW("A client tried to connect, but we already "
187c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                              "have one.");
188d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
189c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        mNetSession->destroySession(sessionID);
190c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        break;
191c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    }
192c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
193c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    CHECK(msg->findString("client-ip", &mClientInfo.mRemoteIP));
194c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    CHECK(msg->findString("server-ip", &mClientInfo.mLocalIP));
195c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
196c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    if (mClientInfo.mRemoteIP == mClientInfo.mLocalIP) {
197c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        // Disallow connections from the local interface
198c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        // for security reasons.
199c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        mNetSession->destroySession(sessionID);
200c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        break;
201c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    }
202d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
203c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    CHECK(msg->findInt32(
204c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                                "server-port", &mClientInfo.mLocalPort));
205c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    mClientInfo.mPlaybackSessionID = -1;
206c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
207c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    mClientSessionID = sessionID;
208c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
209c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    ALOGI("We now have a client (%d) connected.", sessionID);
210d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
211d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    status_t err = sendM1(sessionID);
212d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK_EQ(err, (status_t)OK);
213d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    break;
214d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
215d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
216d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                case ANetworkSession::kWhatData:
217d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                {
218b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    status_t err = onReceiveClientData(msg);
219b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
220b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    if (err != OK) {
221ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                        mClient->onDisplayError(
222ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                                IRemoteDisplayClient::kDisplayErrorUnknown);
223b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    }
224d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    break;
225d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
226d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
227d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                default:
228d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    TRESPASS();
229d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
230d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
231d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
232d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
233d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatStop:
234d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
235eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber            CHECK(msg->senderAwaitsResponse(&mStopReplyID));
236d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
237ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber            if (mClientSessionID != 0
238ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                    && mClientInfo.mPlaybackSessionID != -1) {
239ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber                status_t err = sendM5(
240ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber                        mClientSessionID, true /* requestShutdown */);
2410b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber
242ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber                if (err == OK) {
243ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber                    break;
244ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber                }
245b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            }
246d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
247eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber            finishStop();
248d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
249d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
250d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
251d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatReapDeadClients:
252d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
253d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            mReaperPending = false;
254d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
255c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            if (mClientSessionID == 0
256c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    || mClientInfo.mPlaybackSession == NULL) {
257c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                break;
258d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
259d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
260c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            if (mClientInfo.mPlaybackSession->getLastLifesignUs()
261c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    + kPlaybackSessionTimeoutUs < ALooper::GetNowUs()) {
262c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                ALOGI("playback session timed out, reaping.");
263c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
264ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                mNetSession->destroySession(mClientSessionID);
265ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                mClientSessionID = 0;
266ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber
267ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                mClient->onDisplayError(
268ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                        IRemoteDisplayClient::kDisplayErrorUnknown);
269c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            } else {
270d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                scheduleReaper();
271d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
272d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
273d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
274d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
275d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatPlaybackSessionNotify:
276d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
277d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            int32_t playbackSessionID;
278d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->findInt32("playbackSessionID", &playbackSessionID));
279d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
280d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            int32_t what;
281d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->findInt32("what", &what));
282d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
283c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            if (what == PlaybackSession::kWhatSessionDead) {
284c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                ALOGI("playback session wants to quit.");
285c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
286ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                mClient->onDisplayError(
287ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                        IRemoteDisplayClient::kDisplayErrorUnknown);
288c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            } else if (what == PlaybackSession::kWhatSessionEstablished) {
289c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                if (mClient != NULL) {
290c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                    mClient->onDisplayConnected(
291c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                            mClientInfo.mPlaybackSession->getSurfaceTexture(),
292c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                            mClientInfo.mPlaybackSession->width(),
293c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                            mClientInfo.mPlaybackSession->height(),
294c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                            0 /* flags */);
295c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                }
29696fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber            } else if (what == PlaybackSession::kWhatSessionDestroyed) {
29796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber                disconnectClient2();
298c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            } else {
299c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                CHECK_EQ(what, PlaybackSession::kWhatBinaryData);
300d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
301c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                int32_t channel;
302c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                CHECK(msg->findInt32("channel", &channel));
303d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
304c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                sp<ABuffer> data;
305c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                CHECK(msg->findBuffer("data", &data));
306d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
307c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                CHECK_LE(channel, 0xffu);
308c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                CHECK_LE(data->size(), 0xffffu);
309d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
310c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                int32_t sessionID;
311c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                CHECK(msg->findInt32("sessionID", &sessionID));
312d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
313c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                char header[4];
314c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                header[0] = '$';
315c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                header[1] = channel;
316c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                header[2] = data->size() >> 8;
317c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                header[3] = data->size() & 0xff;
318d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
319c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                mNetSession->sendRequest(
320c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        sessionID, header, sizeof(header));
321d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
322c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                mNetSession->sendRequest(
323c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber                        sessionID, data->data(), data->size());
324d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
325d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
326d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
327d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
328b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        case kWhatKeepAlive:
329b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        {
330b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            int32_t sessionID;
331b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            CHECK(msg->findInt32("sessionID", &sessionID));
332b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
333c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            if (mClientSessionID != sessionID) {
334b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber                // Obsolete event, client is already gone.
335b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber                break;
336b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            }
337b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
338b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            sendM16(sessionID);
339b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            break;
340b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        }
341b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
342b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#if REQUIRE_HDCP
343b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        case kWhatHDCPNotify:
344b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        {
345b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            int32_t msgCode, ext1, ext2;
346b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            CHECK(msg->findInt32("msg", &msgCode));
347b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            CHECK(msg->findInt32("ext1", &ext1));
348b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            CHECK(msg->findInt32("ext2", &ext2));
349b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
350eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber            ALOGI("Saw HDCP notification code %d, ext1 %d, ext2 %d",
351b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    msgCode, ext1, ext2);
352b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
353b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            switch (msgCode) {
354b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                case HDCPModule::HDCP_INITIALIZATION_COMPLETE:
355b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                {
356b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    mHDCPInitializationComplete = true;
357b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
358b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    if (mSetupTriggerDeferred) {
359b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                        mSetupTriggerDeferred = false;
360b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
361ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber                        sendM5(mClientSessionID, false /* requestShutdown */);
362b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    }
363b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    break;
364b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                }
365b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
366eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber                case HDCPModule::HDCP_SHUTDOWN_COMPLETE:
367ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                case HDCPModule::HDCP_SHUTDOWN_FAILED:
368eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber                {
369ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                    // Ugly hack to make sure that the call to
370ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                    // HDCPObserver::notify is completely handled before
371ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                    // we clear the HDCP instance and unload the shared
372ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                    // library :(
373ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                    (new AMessage(kWhatFinishStop2, id()))->post(300000ll);
374eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber                    break;
375eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber                }
376eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber
377b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                default:
378b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                {
379ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber                    ALOGE("HDCP failure, shutting down.");
380ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber
381ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                    mClient->onDisplayError(
382ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber                            IRemoteDisplayClient::kDisplayErrorUnknown);
383b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                    break;
384b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                }
385b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            }
386b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            break;
387b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        }
388ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber
389ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        case kWhatFinishStop2:
390ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        {
391ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber            finishStop2();
392ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber            break;
393ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        }
394b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#endif
395b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
396d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        default:
397d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            TRESPASS();
398d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
399d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
400d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
401d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::registerResponseHandler(
402d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, int32_t cseq, HandleRTSPResponseFunc func) {
403d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ResponseID id;
404d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    id.mSessionID = sessionID;
405d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    id.mCSeq = cseq;
406d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    mResponseHandlers.add(id, func);
407d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
408d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
409d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM1(int32_t sessionID) {
410d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "OPTIONS * RTSP/1.0\r\n";
411d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
412d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
413d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(
414d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "Require: org.wfa.wfd1.0\r\n"
415d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "\r\n");
416d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
417d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
418d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
419d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
420d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
421d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
422d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
423d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
424d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
425d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM1Response);
426d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
427d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
428d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
429d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
430d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
431d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
432d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM3(int32_t sessionID) {
433d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString body =
434b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#if REQUIRE_HDCP
435b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        "wfd_content_protection\r\n"
436b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#endif
437d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_video_formats\r\n"
438d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_audio_codecs\r\n"
439d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_client_rtp_ports\r\n";
440d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
441d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "GET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
442d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
443d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
444d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("Content-Type: text/parameters\r\n");
445d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(StringPrintf("Content-Length: %d\r\n", body.size()));
446d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("\r\n");
447d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(body);
448d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
449d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
450d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
451d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
452d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
453d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
454d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
455d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
456d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
457d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM3Response);
458d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
459d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
460d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
461d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
462d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
463d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
464d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM4(int32_t sessionID) {
465d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // wfd_video_formats:
466d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // 1 byte "native"
467d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // 1 byte "preferred-display-mode-supported" 0 or 1
468d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // one or more avc codec structures
469d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte profile
470d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte level
471d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   4 byte CEA mask
472d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   4 byte VESA mask
473d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   4 byte HH mask
474d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte latency
475d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   2 byte min-slice-slice
476d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   2 byte slice-enc-params
477d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte framerate-control-support
478d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   max-hres (none or 2 byte)
479d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   max-vres (none or 2 byte)
480d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
481c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    CHECK_EQ(sessionID, mClientSessionID);
482d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
483bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber    AString transportString = "UDP";
484bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber
485bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber    char val[PROPERTY_VALUE_MAX];
486bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber    if (property_get("media.wfd.enable-tcp", val, NULL)
487bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            && (!strcasecmp("true", val) || !strcmp("1", val))) {
488bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        ALOGI("Using TCP transport.");
489bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        transportString = "TCP";
490bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber    }
491bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber
492b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    // For 720p60:
493b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    //   use "30 00 02 02 00000040 00000000 00000000 00 0000 0000 00 none none\r\n"
494b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    // For 720p30:
495b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    //   use "28 00 02 02 00000020 00000000 00000000 00 0000 0000 00 none none\r\n"
49672ff5903df5f409ea83f74c363a52f0745ced8b8Andreas Huber    // For 720p24:
49772ff5903df5f409ea83f74c363a52f0745ced8b8Andreas Huber    //   use "78 00 02 02 00008000 00000000 00000000 00 0000 0000 00 none none\r\n"
498d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString body = StringPrintf(
499d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_video_formats: "
50072ff5903df5f409ea83f74c363a52f0745ced8b8Andreas Huber        "78 00 02 02 00008000 00000000 00000000 00 0000 0000 00 none none\r\n"
501d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_audio_codecs: AAC 00000001 00\r\n"  // 2 ch AAC 48kHz
502de799a74064a363d26f4c1bbc5a59d1b7127f49fAndreas Huber        "wfd_presentation_URL: rtsp://%s/wfd1.0/streamid=0 none\r\n"
503bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        "wfd_client_rtp_ports: RTP/AVP/%s;unicast 19000 0 mode=play\r\n",
504de799a74064a363d26f4c1bbc5a59d1b7127f49fAndreas Huber        mClientInfo.mLocalIP.c_str(), transportString.c_str());
505d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
506d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "SET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
507d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
508d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
509d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("Content-Type: text/parameters\r\n");
510d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(StringPrintf("Content-Length: %d\r\n", body.size()));
511d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("\r\n");
512d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(body);
513d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
514d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
515d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
516d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
517d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
518d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
519d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
520d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
521d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
522d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM4Response);
523d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
524d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
525d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
526d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
527d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
528d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
529ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huberstatus_t WifiDisplaySource::sendM5(int32_t sessionID, bool requestShutdown) {
530ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    AString body = "wfd_trigger_method: ";
531ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    if (requestShutdown) {
532ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        ALOGI("Sending TEARDOWN trigger.");
533ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber        body.append("TEARDOWN");
534ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    } else {
535ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber        body.append("SETUP");
536ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    }
537ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber
538ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    body.append("\r\n");
539d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
540d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "SET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
541d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
542d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
543d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("Content-Type: text/parameters\r\n");
544d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(StringPrintf("Content-Length: %d\r\n", body.size()));
545d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("\r\n");
546d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(body);
547d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
548d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
549d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
550d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
551d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
552d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
553d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
554d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
555d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
556d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM5Response);
557d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
558d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
559d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
560d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
561d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
562d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
563b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huberstatus_t WifiDisplaySource::sendM16(int32_t sessionID) {
564b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    AString request = "GET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
565b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
566b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
567c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    CHECK_EQ(sessionID, mClientSessionID);
568c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    request.append(
569c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            StringPrintf("Session: %d\r\n", mClientInfo.mPlaybackSessionID));
570a438123bd96c7faf145683876702387efe5628d9Andreas Huber    request.append("\r\n");  // Empty body
571b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
572b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    status_t err =
573b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
574b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
575b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    if (err != OK) {
576b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        return err;
577b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    }
578b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
579b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    registerResponseHandler(
580b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM16Response);
581b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
582b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    ++mNextCSeq;
583b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
584b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    return OK;
585b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber}
586b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
587d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM1Response(
588d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
589d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
590d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
591d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
592d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
593d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
594d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
595d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
596d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
597d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
598d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
599d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
600d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
601d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM3Response(
602d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
603d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
604d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
605d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
606d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
607d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
608d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
609d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
610d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
611d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
612b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    sp<Parameters> params =
613b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        Parameters::Parse(msg->getContent(), strlen(msg->getContent()));
614b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
615b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (params == NULL) {
616b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
617b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
618b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
619b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#if REQUIRE_HDCP
620b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    AString value;
621b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (!params->findParameter("wfd_content_protection", &value)) {
622b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        ALOGE("Sink doesn't appear to support content protection.");
623b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return -EACCES;
624b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
625b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
626b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (value == "none") {
627b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        ALOGE("Sink does not support content protection.");
628b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return -EACCES;
629b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
630b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
631b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    bool isHDCP2_0 = false;
632b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (value.startsWith("HDCP2.0 ")) {
633b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        isHDCP2_0 = true;
634b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else if (!value.startsWith("HDCP2.1 ")) {
635b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
636b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
637b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
638b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    int32_t hdcpPort;
639b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (!ParsedMessage::GetInt32Attribute(value.c_str() + 8, "port", &hdcpPort)
640b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            || hdcpPort < 1 || hdcpPort > 65535) {
641b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
642b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
643b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
644b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    mIsHDCP2_0 = isHDCP2_0;
645b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    mHDCPPort = hdcpPort;
646b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
647b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    status_t err = makeHDCP();
648b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (err != OK) {
649b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        ALOGE("Unable to instantiate HDCP component.");
650b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return err;
651b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
652b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#endif
653b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
654d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return sendM4(sessionID);
655d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
656d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
657d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM4Response(
658d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
659d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
660d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
661d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
662d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
663d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
664d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
665d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
666d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
667d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
668b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#if REQUIRE_HDCP
669b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (!mHDCPInitializationComplete) {
670b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        ALOGI("Deferring SETUP trigger until HDCP initialization completes.");
671b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
672b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        mSetupTriggerDeferred = true;
673b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return OK;
674b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
675b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#endif
676b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
677ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    return sendM5(sessionID, false /* requestShutdown */);
678d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
679d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
680d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM5Response(
681d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
682d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
683d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
684d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
685d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
686d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
687d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
688d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
689d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
690d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
691d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
692d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
693d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
694b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huberstatus_t WifiDisplaySource::onReceiveM16Response(
695b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
696b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    // If only the response was required to include a "Session:" header...
697b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
698c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    CHECK_EQ(sessionID, mClientSessionID);
699b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
700c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    if (mClientInfo.mPlaybackSession != NULL) {
701c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber        mClientInfo.mPlaybackSession->updateLiveness();
702b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
703b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        scheduleKeepAlive(sessionID);
704b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    }
705b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
706b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    return OK;
707b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber}
708b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
709d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::scheduleReaper() {
710d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (mReaperPending) {
711d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
712d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
713d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
714d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    mReaperPending = true;
715d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    (new AMessage(kWhatReapDeadClients, id()))->post(kReaperIntervalUs);
716d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
717d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
718b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Hubervoid WifiDisplaySource::scheduleKeepAlive(int32_t sessionID) {
719b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    // We need to send updates at least 5 secs before the timeout is set to
720b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    // expire, make sure the timeout is greater than 5 secs to begin with.
721b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    CHECK_GT(kPlaybackSessionTimeoutUs, 5000000ll);
722b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
723b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    sp<AMessage> msg = new AMessage(kWhatKeepAlive, id());
724b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    msg->setInt32("sessionID", sessionID);
725b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    msg->post(kPlaybackSessionTimeoutUs - 5000000ll);
726b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber}
727b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
728b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onReceiveClientData(const sp<AMessage> &msg) {
729d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t sessionID;
730d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK(msg->findInt32("sessionID", &sessionID));
731d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
732d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<RefBase> obj;
733d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK(msg->findObject("data", &obj));
734d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
735d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<ParsedMessage> data =
736d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        static_cast<ParsedMessage *>(obj.get());
737d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
738d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ALOGV("session %d received '%s'",
739d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber          sessionID, data->debugString().c_str());
740d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
741d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString method;
742d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString uri;
743d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    data->getRequestField(0, &method);
744d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
745d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t cseq;
746d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!data->findInt32("cseq", &cseq)) {
747d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", -1 /* cseq */);
748b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
749d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
750d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
751d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (method.startsWith("RTSP/")) {
752d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        // This is a response.
753d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
754d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        ResponseID id;
755d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        id.mSessionID = sessionID;
756d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        id.mCSeq = cseq;
757d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
758d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        ssize_t index = mResponseHandlers.indexOfKey(id);
759d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
760d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (index < 0) {
761d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            ALOGW("Received unsolicited server response, cseq %d", cseq);
762b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            return ERROR_MALFORMED;
763d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
764d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
765d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        HandleRTSPResponseFunc func = mResponseHandlers.valueAt(index);
766d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mResponseHandlers.removeItemsAt(index);
767d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
768d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        status_t err = (this->*func)(sessionID, data);
769d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
770d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (err != OK) {
771d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            ALOGW("Response handler for session %d, cseq %d returned "
772d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                  "err %d (%s)",
773d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                  sessionID, cseq, err, strerror(-err));
774d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
775b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            return err;
776d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
777d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
778b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return OK;
779b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
780d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
781b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    AString version;
782b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    data->getRequestField(2, &version);
783b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (!(version == AString("RTSP/1.0"))) {
784b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        sendErrorResponse(sessionID, "505 RTSP Version not supported", cseq);
785b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_UNSUPPORTED;
786b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
787d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
788b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    status_t err;
789b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (method == "OPTIONS") {
790b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = onOptionsRequest(sessionID, cseq, data);
791b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else if (method == "SETUP") {
792b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = onSetupRequest(sessionID, cseq, data);
793b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else if (method == "PLAY") {
794b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = onPlayRequest(sessionID, cseq, data);
795b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else if (method == "PAUSE") {
796b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = onPauseRequest(sessionID, cseq, data);
797b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else if (method == "TEARDOWN") {
798b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = onTeardownRequest(sessionID, cseq, data);
799b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else if (method == "GET_PARAMETER") {
800b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = onGetParameterRequest(sessionID, cseq, data);
801b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else if (method == "SET_PARAMETER") {
802b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = onSetParameterRequest(sessionID, cseq, data);
803b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    } else {
804b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        sendErrorResponse(sessionID, "405 Method Not Allowed", cseq);
805d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
806b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = ERROR_UNSUPPORTED;
807b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
808d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
809b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return err;
810d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
811d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
812b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onOptionsRequest(
813d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
814d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
815d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
816d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
817d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
818d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
819d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
820d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession != NULL) {
821d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        playbackSession->updateLiveness();
822d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
823d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
824d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
825d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq);
826d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
827d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append(
828b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            "Public: org.wfa.wfd1.0, SETUP, TEARDOWN, PLAY, PAUSE, "
829d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "GET_PARAMETER, SET_PARAMETER\r\n");
830d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
831d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
832d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
833d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
834d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
835b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (err == OK) {
836b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        err = sendM3(sessionID);
837b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
838b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
839b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return err;
840d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
841d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
842b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onSetupRequest(
843d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
844d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
845d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
846c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    CHECK_EQ(sessionID, mClientSessionID);
847c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    if (mClientInfo.mPlaybackSessionID != -1) {
848b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        // We only support a single playback session per client.
849b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        // This is due to the reversed keep-alive design in the wfd specs...
850b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", cseq);
851b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
852b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    }
853b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
854d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString transport;
855d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!data->findString("transport", &transport)) {
856d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", cseq);
857b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
858d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
859d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
860bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber    PlaybackSession::TransportMode transportMode =
861bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        PlaybackSession::TRANSPORT_UDP;
862d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
863d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int clientRtp, clientRtcp;
864d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (transport.startsWith("RTP/AVP/TCP;")) {
865d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        AString interleaved;
866bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        if (ParsedMessage::GetAttribute(
867d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    transport.c_str(), "interleaved", &interleaved)
868bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                && sscanf(interleaved.c_str(), "%d-%d",
869bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                          &clientRtp, &clientRtcp) == 2) {
870bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            transportMode = PlaybackSession::TRANSPORT_TCP_INTERLEAVED;
871bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        } else {
872bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            bool badRequest = false;
873bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber
874bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            AString clientPort;
875bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            if (!ParsedMessage::GetAttribute(
876bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                        transport.c_str(), "client_port", &clientPort)) {
877bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                badRequest = true;
878bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            } else if (sscanf(clientPort.c_str(), "%d-%d",
879bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                              &clientRtp, &clientRtcp) == 2) {
880bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            } else if (sscanf(clientPort.c_str(), "%d", &clientRtp) == 1) {
881bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                // No RTCP.
882bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                clientRtcp = -1;
883bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            } else {
884bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                badRequest = true;
885bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            }
886bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber
887bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            if (badRequest) {
888bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                sendErrorResponse(sessionID, "400 Bad Request", cseq);
889b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                return ERROR_MALFORMED;
890bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            }
891d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
892bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            transportMode = PlaybackSession::TRANSPORT_TCP;
893bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        }
894d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else if (transport.startsWith("RTP/AVP;unicast;")
895d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            || transport.startsWith("RTP/AVP/UDP;unicast;")) {
896d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        bool badRequest = false;
897d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
898d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        AString clientPort;
899d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (!ParsedMessage::GetAttribute(
900d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    transport.c_str(), "client_port", &clientPort)) {
901d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            badRequest = true;
902d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (sscanf(clientPort.c_str(), "%d-%d",
903d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          &clientRtp, &clientRtcp) == 2) {
904d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (sscanf(clientPort.c_str(), "%d", &clientRtp) == 1) {
905d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            // No RTCP.
906d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            clientRtcp = -1;
907d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else {
908d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            badRequest = true;
909d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
910d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
911d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (badRequest) {
912d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "400 Bad Request", cseq);
913b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            return ERROR_MALFORMED;
914d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
915d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#if 1
916a438123bd96c7faf145683876702387efe5628d9Andreas Huber    // The older LG dongles doesn't specify client_port=xxx apparently.
917d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else if (transport == "RTP/AVP/UDP;unicast") {
918d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        clientRtp = 19000;
91928e17ed7e2fbb254fb99481b74db85e427c905eeAndreas Huber        clientRtcp = -1;
920d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#endif
921d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else {
922d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "461 Unsupported Transport", cseq);
923b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_UNSUPPORTED;
924d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
925d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
926d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID = makeUniquePlaybackSessionID();
927d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
928d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> notify = new AMessage(kWhatPlaybackSessionNotify, id());
929d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    notify->setInt32("playbackSessionID", playbackSessionID);
930d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    notify->setInt32("sessionID", sessionID);
931d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
932d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
9330b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber        new PlaybackSession(
934bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                mNetSession, notify, mInterfaceAddr,
935b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#if REQUIRE_HDCP
936b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                mHDCP
937b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#else
938b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                NULL
939b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#endif
940b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber                );
941d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
942d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    looper()->registerHandler(playbackSession);
943d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
944d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString uri;
945d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    data->getRequestField(1, &uri);
946d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
947d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (strncasecmp("rtsp://", uri.c_str(), 7)) {
948d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", cseq);
949b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
950d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
951d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
952d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!(uri.startsWith("rtsp://") && uri.endsWith("/wfd1.0/streamid=0"))) {
953d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "404 Not found", cseq);
954b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
955d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
956d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
957d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = playbackSession->init(
958c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber            mClientInfo.mRemoteIP.c_str(),
959d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            clientRtp,
960d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            clientRtcp,
961bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            transportMode);
962d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
963d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
964d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        looper()->unregisterHandler(playbackSession->id());
965d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        playbackSession.clear();
966d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
967d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
968d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    switch (err) {
969d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case OK:
970d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
971d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case -ENOENT:
972d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "404 Not Found", cseq);
973b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            return err;
974d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        default:
975d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "403 Forbidden", cseq);
976b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            return err;
977d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
978d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
979c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    mClientInfo.mPlaybackSessionID = playbackSessionID;
980c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    mClientInfo.mPlaybackSession = playbackSession;
981b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
982d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
983d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
984d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
985bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber    if (transportMode == PlaybackSession::TRANSPORT_TCP_INTERLEAVED) {
986d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        response.append(
987d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                StringPrintf(
988d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    "Transport: RTP/AVP/TCP;interleaved=%d-%d;",
989d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    clientRtp, clientRtcp));
990d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else {
991d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t serverRtp = playbackSession->getRTPPort();
992d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
993bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        AString transportString = "UDP";
994bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        if (transportMode == PlaybackSession::TRANSPORT_TCP) {
995bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber            transportString = "TCP";
996bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber        }
997bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber
998d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (clientRtcp >= 0) {
999d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response.append(
1000d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    StringPrintf(
1001bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                        "Transport: RTP/AVP/%s;unicast;client_port=%d-%d;"
1002d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        "server_port=%d-%d\r\n",
1003bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                        transportString.c_str(),
1004d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        clientRtp, clientRtcp, serverRtp, serverRtp + 1));
1005d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else {
1006d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response.append(
1007d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    StringPrintf(
1008bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                        "Transport: RTP/AVP/%s;unicast;client_port=%d;"
1009d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        "server_port=%d\r\n",
1010bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber                        transportString.c_str(),
1011d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        clientRtp, serverRtp));
1012d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
1013d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1014d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1015d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1016d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1017d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    err = mNetSession->sendRequest(sessionID, response.c_str());
1018b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1019b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (err != OK) {
1020b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return err;
1021b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
1022d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1023d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    scheduleReaper();
1024b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    scheduleKeepAlive(sessionID);
1025b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1026b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return OK;
1027d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1028d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1029b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onPlayRequest(
1030d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
1031d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
1032d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
1033d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
1034d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
1035d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
1036d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1037d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
1038d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
1039b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
1040d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1041d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1042ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    ALOGI("Received PLAY request.");
1043ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber
1044d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = playbackSession->play();
1045d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
1046d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1047d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
1048d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
1049d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("Range: npt=now-\r\n");
1050d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1051d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1052d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    err = mNetSession->sendRequest(sessionID, response.c_str());
1053b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1054b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (err != OK) {
1055b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return err;
1056b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
10570b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber
1058bd08e2f93bafd02abf2c25d740e9fb8bce455a99Andreas Huber    playbackSession->finishPlay();
1059b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1060b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return OK;
1061d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1062d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1063b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onPauseRequest(
1064d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
1065d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
1066d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
1067d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
1068d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
1069d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
1070d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1071d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
1072d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
1073b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
1074d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1075d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1076d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = playbackSession->pause();
1077d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
1078d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1079d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
1080d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
1081d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1082d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1083d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    err = mNetSession->sendRequest(sessionID, response.c_str());
1084b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1085b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return err;
1086d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1087d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1088b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onTeardownRequest(
1089d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
1090d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
1091d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
1092ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    ALOGI("Received TEARDOWN request.");
1093ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber
1094d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
1095d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
1096d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
1097d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1098d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
1099d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
1100b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
1101d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1102d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1103d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
1104d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
1105c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    response.append("Connection: close\r\n");
1106d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1107d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1108ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    mNetSession->sendRequest(sessionID, response.c_str());
1109c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
1110ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    if (mStopReplyID != 0) {
1111eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber        finishStop();
1112ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    } else {
1113ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        mClient->onDisplayError(IRemoteDisplayClient::kDisplayErrorUnknown);
1114ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    }
1115b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1116b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return OK;
1117d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1118d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1119eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Hubervoid WifiDisplaySource::finishStop() {
1120ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    ALOGV("finishStop");
1121ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber
112296fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    disconnectClientAsync();
112396fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber}
112496fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber
112596fc6cc65ca93009a759a3a874b82a35771b9714Andreas Hubervoid WifiDisplaySource::finishStopAfterDisconnectingClient() {
112696fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    ALOGV("finishStopAfterDisconnectingClient");
112796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber
1128ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber#if REQUIRE_HDCP
1129ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    if (mHDCP != NULL) {
1130ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        ALOGI("Initiating HDCP shutdown.");
1131ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber        mHDCP->shutdownAsync();
1132eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber        return;
1133ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    }
1134ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber#endif
1135ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber
1136eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber    finishStop2();
1137eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber}
1138eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber
1139eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Hubervoid WifiDisplaySource::finishStop2() {
1140ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    ALOGV("finishStop2");
1141ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber
1142eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber#if REQUIRE_HDCP
1143ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    mHDCP->setObserver(NULL);
1144ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    mHDCPObserver.clear();
1145eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber    mHDCP.clear();
1146eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber#endif
1147eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber
1148ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    if (mSessionID != 0) {
1149ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        mNetSession->destroySession(mSessionID);
1150ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        mSessionID = 0;
1151ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    }
1152ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber
115396fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    ALOGI("We're stopped.");
1154ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber
1155ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    status_t err = OK;
1156ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber
1157ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    sp<AMessage> response = new AMessage;
1158ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber    response->setInt32("err", err);
1159eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber    response->postReply(mStopReplyID);
1160ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber}
1161ea4bbfdcad9478ea19257fb19a32de68a2dfd958Andreas Huber
1162b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onGetParameterRequest(
1163d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
1164d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
1165d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
1166d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
1167d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
1168d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
1169d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1170d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
1171d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
1172b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
1173d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1174d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1175d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    playbackSession->updateLiveness();
1176d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1177d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
1178d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
1179d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1180d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1181d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
1182b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return err;
1183d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1184d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1185b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::onSetParameterRequest(
1186d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
1187d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
1188d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
1189d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
1190d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
1191d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
1192d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1193d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
1194d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
1195b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_MALFORMED;
1196d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1197d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1198b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (strstr(data->getContent(), "wfd_idr_request\r\n")) {
1199b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        playbackSession->requestIDRFrame();
1200b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
1201496238cc7551d414067dcbbb4fe3bd801f205f95Andreas Huber
1202d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    playbackSession->updateLiveness();
1203d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1204d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
1205d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
1206d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1207d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1208d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
1209b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return err;
1210d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1211d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1212d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber// static
1213d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::AppendCommonResponse(
1214d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        AString *response, int32_t cseq, int32_t playbackSessionID) {
1215d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    time_t now = time(NULL);
1216d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    struct tm *now2 = gmtime(&now);
1217d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    char buf[128];
1218d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    strftime(buf, sizeof(buf), "%a, %d %b %Y %H:%M:%S %z", now2);
1219d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1220d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append("Date: ");
1221d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append(buf);
1222d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append("\r\n");
1223d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1224d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append("Server: Mine/1.0\r\n");
1225d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1226d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (cseq >= 0) {
1227d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        response->append(StringPrintf("CSeq: %d\r\n", cseq));
1228d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1229d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1230d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSessionID >= 0ll) {
1231d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        response->append(
1232d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                StringPrintf(
1233d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    "Session: %d;timeout=%lld\r\n",
1234d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    playbackSessionID, kPlaybackSessionTimeoutSecs));
1235d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1236d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1237d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1238d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::sendErrorResponse(
1239d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
1240d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const char *errorDetail,
1241d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq) {
1242d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response;
1243d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("RTSP/1.0 ");
1244d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append(errorDetail);
1245d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1246d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1247d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq);
1248d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1249d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1250d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1251eb11600a248cfe5b95ddd3e5aaae02bd2ab65276Andreas Huber    mNetSession->sendRequest(sessionID, response.c_str());
1252d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1253d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1254d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberint32_t WifiDisplaySource::makeUniquePlaybackSessionID() const {
1255c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    return rand();
1256d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1257d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1258d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubersp<WifiDisplaySource::PlaybackSession> WifiDisplaySource::findPlaybackSession(
1259d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data, int32_t *playbackSessionID) const {
1260d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!data->findInt32("session", playbackSessionID)) {
1261c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber        // XXX the older dongles do not always include a "Session:" header.
1262c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber        *playbackSessionID = mClientInfo.mPlaybackSessionID;
1263c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber        return mClientInfo.mPlaybackSession;
1264d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1265d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1266c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    if (*playbackSessionID != mClientInfo.mPlaybackSessionID) {
1267d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return NULL;
1268d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1269d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1270c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    return mClientInfo.mPlaybackSession;
1271c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber}
1272c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
127396fc6cc65ca93009a759a3a874b82a35771b9714Andreas Hubervoid WifiDisplaySource::disconnectClientAsync() {
127496fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    ALOGV("disconnectClient");
127596fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber
127696fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    if (mClientInfo.mPlaybackSession == NULL) {
127796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber        disconnectClient2();
127896fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber        return;
127996fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    }
128096fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber
128196fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    if (mClientInfo.mPlaybackSession != NULL) {
128296fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber        ALOGV("Destroying PlaybackSession");
128396fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber        mClientInfo.mPlaybackSession->destroyAsync();
128496fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    }
128596fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber}
128696fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber
128796fc6cc65ca93009a759a3a874b82a35771b9714Andreas Hubervoid WifiDisplaySource::disconnectClient2() {
128896fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    ALOGV("disconnectClient2");
128996fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber
1290ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    if (mClientInfo.mPlaybackSession != NULL) {
129196fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber        looper()->unregisterHandler(mClientInfo.mPlaybackSession->id());
1292ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber        mClientInfo.mPlaybackSession.clear();
1293ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    }
1294c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
1295ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    if (mClientSessionID != 0) {
1296c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber        mNetSession->destroySession(mClientSessionID);
1297c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber        mClientSessionID = 0;
1298c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber    }
1299c92bed3a73c06e90217f8f199ca0b517aa7595d2Andreas Huber
1300ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    mClient->onDisplayDisconnected();
130196fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber
130296fc6cc65ca93009a759a3a874b82a35771b9714Andreas Huber    finishStopAfterDisconnectingClient();
1303d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1304d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1305b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#if REQUIRE_HDCP
1306b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstruct WifiDisplaySource::HDCPObserver : public BnHDCPObserver {
1307b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    HDCPObserver(const sp<AMessage> &notify);
1308b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1309b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    virtual void notify(
1310b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            int msg, int ext1, int ext2, const Parcel *obj);
1311b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1312b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberprivate:
1313b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    sp<AMessage> mNotify;
1314b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1315b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    DISALLOW_EVIL_CONSTRUCTORS(HDCPObserver);
1316b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber};
1317b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1318b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas HuberWifiDisplaySource::HDCPObserver::HDCPObserver(
1319b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        const sp<AMessage> &notify)
1320b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    : mNotify(notify) {
1321b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber}
1322b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1323b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Hubervoid WifiDisplaySource::HDCPObserver::notify(
1324b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        int msg, int ext1, int ext2, const Parcel *obj) {
1325b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    sp<AMessage> notify = mNotify->dup();
1326b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    notify->setInt32("msg", msg);
1327b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    notify->setInt32("ext1", ext1);
1328b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    notify->setInt32("ext2", ext2);
1329b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    notify->post();
1330b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber}
1331b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1332b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huberstatus_t WifiDisplaySource::makeHDCP() {
1333b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    sp<IServiceManager> sm = defaultServiceManager();
1334b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    sp<IBinder> binder = sm->getService(String16("media.player"));
1335b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    sp<IMediaPlayerService> service = interface_cast<IMediaPlayerService>(binder);
1336b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    CHECK(service != NULL);
1337b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1338b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    mHDCP = service->makeHDCP();
1339b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1340b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (mHDCP == NULL) {
1341b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return ERROR_UNSUPPORTED;
1342b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
1343b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1344b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    sp<AMessage> notify = new AMessage(kWhatHDCPNotify, id());
1345b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    mHDCPObserver = new HDCPObserver(notify);
1346b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1347b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    status_t err = mHDCP->setObserver(mHDCPObserver);
1348b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1349b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (err != OK) {
1350b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        ALOGE("Failed to set HDCP observer.");
1351b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1352b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        mHDCPObserver.clear();
1353b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        mHDCP.clear();
1354b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1355b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return err;
1356b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
1357b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1358ef7d3793fa9bbfb25253626ede9a020ee9280a17Andreas Huber    ALOGI("Initiating HDCP negotiation w/ host %s:%d",
1359b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber            mClientInfo.mRemoteIP.c_str(), mHDCPPort);
1360b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1361b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    err = mHDCP->initAsync(mClientInfo.mRemoteIP.c_str(), mHDCPPort);
1362b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1363b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    if (err != OK) {
1364b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber        return err;
1365b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    }
1366b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1367b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber    return OK;
1368b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber}
1369b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber#endif
1370b8c7bd418f0ee5b88923b0e0817e3a4acc53cf8dAndreas Huber
1371d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}  // namespace android
1372d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1373