WifiDisplaySource.cpp revision a438123bd96c7faf145683876702387efe5628d9
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"
23d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include "ParsedMessage.h"
24d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
250b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber#include <gui/ISurfaceTexture.h>
260b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber
270b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber#include <media/IRemoteDisplayClient.h>
28d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/ABuffer.h>
29d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/ADebug.h>
30d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/foundation/AMessage.h>
31d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#include <media/stagefright/MediaErrors.h>
32d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
33bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber#include <arpa/inet.h>
34bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber#include <netinet/in.h>
35bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
36d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubernamespace android {
37d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
380b73d4730202fcad53aefc4314a06e7b95f442f0Andreas HuberWifiDisplaySource::WifiDisplaySource(
390b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber        const sp<ANetworkSession> &netSession,
400b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber        const sp<IRemoteDisplayClient> &client)
41d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    : mNetSession(netSession),
420b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber      mClient(client),
43d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber      mSessionID(0),
44d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber      mReaperPending(false),
45d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber      mNextCSeq(1) {
46d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
47d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
48d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas HuberWifiDisplaySource::~WifiDisplaySource() {
49d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
50d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
51bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huberstatus_t WifiDisplaySource::start(const char *iface) {
52d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> msg = new AMessage(kWhatStart, id());
53bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber    msg->setString("iface", iface);
54d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
55d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> response;
56d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = msg->postAndAwaitResponse(&response);
57d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
58d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
59d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
60d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
61d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
62d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!response->findInt32("err", &err)) {
63d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        err = OK;
64d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
65d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
66d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return err;
67d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
68d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
69d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::stop() {
70d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> msg = new AMessage(kWhatStop, id());
71d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
72d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> response;
73d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = msg->postAndAwaitResponse(&response);
74d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
75d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
76d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
77d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
78d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
79d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!response->findInt32("err", &err)) {
80d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        err = OK;
81d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
82d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
83d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return err;
84d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
85d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
86d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onMessageReceived(const sp<AMessage> &msg) {
87d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    switch (msg->what()) {
88d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatStart:
89d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
90d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            uint32_t replyID;
91d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->senderAwaitsResponse(&replyID));
92d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
93bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            AString iface;
94bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            CHECK(msg->findString("iface", &iface));
95bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
96bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            status_t err = OK;
97bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
98bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            ssize_t colonPos = iface.find(":");
99bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
100bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            unsigned long port;
101bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
102bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            if (colonPos >= 0) {
103bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                const char *s = iface.c_str() + colonPos + 1;
104bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
105bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                char *end;
106bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                port = strtoul(s, &end, 10);
107bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
108bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                if (end == s || *end != '\0' || port > 65535) {
109bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    err = -EINVAL;
110bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                } else {
111bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    iface.erase(colonPos, iface.size() - colonPos);
112bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                }
113bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            } else {
114bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                port = kWifiDisplayDefaultPort;
115bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            }
116bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
117bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            struct in_addr addr;
118bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber
119bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            if (err == OK) {
120bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                if (inet_aton(iface.c_str(), &addr) != 0) {
121bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    sp<AMessage> notify = new AMessage(kWhatRTSPNotify, id());
122d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
123bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    err = mNetSession->createRTSPServer(
124bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                            addr, port, notify, &mSessionID);
125bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                } else {
126bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                    err = -EINVAL;
127bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber                }
128bcf09f8c995221e75c7cd328f25c7cc6d2b5f7c9Andreas Huber            }
129d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
130d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sp<AMessage> response = new AMessage;
131d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response->setInt32("err", err);
132d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response->postReply(replyID);
133d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
134d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
135d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
136d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatRTSPNotify:
137d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
138d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            int32_t reason;
139d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->findInt32("reason", &reason));
140d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
141d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            switch (reason) {
142d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                case ANetworkSession::kWhatError:
143d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                {
144d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t sessionID;
145d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("sessionID", &sessionID));
146d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
147d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t err;
148d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("err", &err));
149d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
150d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    AString detail;
151d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findString("detail", &detail));
152d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
153d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    ALOGE("An error occurred in session %d (%d, '%s/%s').",
154d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          sessionID,
155d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          err,
156d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          detail.c_str(),
157d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          strerror(-err));
158d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
159d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mNetSession->destroySession(sessionID);
160d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
161b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber                    mClientInfos.removeItem(sessionID);
162d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    break;
163d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
164d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
165d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                case ANetworkSession::kWhatClientConnected:
166d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                {
167d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t sessionID;
168d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("sessionID", &sessionID));
169d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
170d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    ClientInfo info;
171d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findString("client-ip", &info.mRemoteIP));
172d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findString("server-ip", &info.mLocalIP));
173d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("server-port", &info.mLocalPort));
174b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber                    info.mPlaybackSessionID = -1;
175d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
176d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    ALOGI("We now have a client (%d) connected.", sessionID);
177d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
178b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber                    mClientInfos.add(sessionID, info);
179d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
180d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    status_t err = sendM1(sessionID);
181d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK_EQ(err, (status_t)OK);
182d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    break;
183d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
184d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
185d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                case ANetworkSession::kWhatData:
186d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                {
187d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    onReceiveClientData(msg);
188d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    break;
189d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
190d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
191d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                default:
192d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    TRESPASS();
193d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
194d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
195d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
196d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
197d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatStop:
198d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
199d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            uint32_t replyID;
200d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->senderAwaitsResponse(&replyID));
201d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
202d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            for (size_t i = mPlaybackSessions.size(); i-- > 0;) {
203a438123bd96c7faf145683876702387efe5628d9Andreas Huber                sp<PlaybackSession> playbackSession =
204d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mPlaybackSessions.valueAt(i);
205d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
206d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                mPlaybackSessions.removeItemsAt(i);
207a438123bd96c7faf145683876702387efe5628d9Andreas Huber
208a438123bd96c7faf145683876702387efe5628d9Andreas Huber                playbackSession->destroy();
209a438123bd96c7faf145683876702387efe5628d9Andreas Huber                looper()->unregisterHandler(playbackSession->id());
210a438123bd96c7faf145683876702387efe5628d9Andreas Huber                playbackSession.clear();
211d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
212d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
2130b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber            if (mClient != NULL) {
2140b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber                mClient->onDisplayDisconnected();
2150b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber            }
2160b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber
217d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            status_t err = OK;
218d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
219d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sp<AMessage> response = new AMessage;
220d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response->setInt32("err", err);
221d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response->postReply(replyID);
222d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
223d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
224d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
225d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatReapDeadClients:
226d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
227d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            mReaperPending = false;
228d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
229d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            for (size_t i = mPlaybackSessions.size(); i-- > 0;) {
230d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                const sp<PlaybackSession> &playbackSession =
231d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mPlaybackSessions.valueAt(i);
232d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
233d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                if (playbackSession->getLastLifesignUs()
234d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        + kPlaybackSessionTimeoutUs < ALooper::GetNowUs()) {
235d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    ALOGI("playback session %d timed out, reaping.",
236d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                            mPlaybackSessions.keyAt(i));
237d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
238d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    looper()->unregisterHandler(playbackSession->id());
239d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mPlaybackSessions.removeItemsAt(i);
240d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
241d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
242d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
243d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            if (!mPlaybackSessions.isEmpty()) {
244d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                scheduleReaper();
245d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
246d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
247d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
248d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
249d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case kWhatPlaybackSessionNotify:
250d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        {
251d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            int32_t playbackSessionID;
252d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->findInt32("playbackSessionID", &playbackSessionID));
253d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
254d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            int32_t what;
255d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            CHECK(msg->findInt32("what", &what));
256d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
257d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            ssize_t index = mPlaybackSessions.indexOfKey(playbackSessionID);
258d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            if (index >= 0) {
259d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                const sp<PlaybackSession> &playbackSession =
260d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mPlaybackSessions.valueAt(index);
261d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
262d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                if (what == PlaybackSession::kWhatSessionDead) {
263d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    ALOGI("playback sessions %d wants to quit.",
264d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          playbackSessionID);
265d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
266d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    looper()->unregisterHandler(playbackSession->id());
267d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mPlaybackSessions.removeItemsAt(index);
268d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                } else {
269d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK_EQ(what, PlaybackSession::kWhatBinaryData);
270d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
271d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t channel;
272d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("channel", &channel));
273d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
274d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    sp<ABuffer> data;
275d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findBuffer("data", &data));
276d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
277d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK_LE(channel, 0xffu);
278d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK_LE(data->size(), 0xffffu);
279d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
280d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    int32_t sessionID;
281d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    CHECK(msg->findInt32("sessionID", &sessionID));
282d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
283d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    char header[4];
284d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    header[0] = '$';
285d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    header[1] = channel;
286d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    header[2] = data->size() >> 8;
287d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    header[3] = data->size() & 0xff;
288d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
289d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mNetSession->sendRequest(
290d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                            sessionID, header, sizeof(header));
291d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
292d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    mNetSession->sendRequest(
293d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                            sessionID, data->data(), data->size());
294d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                }
295d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
296d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
297d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
298d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
299b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        case kWhatKeepAlive:
300b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        {
301b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            int32_t sessionID;
302b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            CHECK(msg->findInt32("sessionID", &sessionID));
303b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
304b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            if (mClientInfos.indexOfKey(sessionID) < 0) {
305b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber                // Obsolete event, client is already gone.
306b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber                break;
307b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            }
308b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
309b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            sendM16(sessionID);
310b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            break;
311b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        }
312b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
313d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        default:
314d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            TRESPASS();
315d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
316d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
317d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
318d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::registerResponseHandler(
319d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, int32_t cseq, HandleRTSPResponseFunc func) {
320d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ResponseID id;
321d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    id.mSessionID = sessionID;
322d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    id.mCSeq = cseq;
323d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    mResponseHandlers.add(id, func);
324d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
325d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
326d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM1(int32_t sessionID) {
327d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "OPTIONS * RTSP/1.0\r\n";
328d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
329d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
330d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(
331d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "Require: org.wfa.wfd1.0\r\n"
332d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "\r\n");
333d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
334d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
335d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
336d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
337d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
338d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
339d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
340d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
341d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
342d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM1Response);
343d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
344d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
345d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
346d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
347d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
348d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
349d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM3(int32_t sessionID) {
350d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString body =
351d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_video_formats\r\n"
352d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_audio_codecs\r\n"
353d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_client_rtp_ports\r\n";
354d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
355d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "GET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
356d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
357d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
358d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("Content-Type: text/parameters\r\n");
359d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(StringPrintf("Content-Length: %d\r\n", body.size()));
360d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("\r\n");
361d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(body);
362d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
363d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
364d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
365d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
366d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
367d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
368d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
369d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
370d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
371d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM3Response);
372d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
373d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
374d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
375d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
376d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
377d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
378d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM4(int32_t sessionID) {
379d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // wfd_video_formats:
380d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // 1 byte "native"
381d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // 1 byte "preferred-display-mode-supported" 0 or 1
382d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // one or more avc codec structures
383d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte profile
384d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte level
385d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   4 byte CEA mask
386d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   4 byte VESA mask
387d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   4 byte HH mask
388d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte latency
389d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   2 byte min-slice-slice
390d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   2 byte slice-enc-params
391d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   1 byte framerate-control-support
392d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   max-hres (none or 2 byte)
393d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    //   max-vres (none or 2 byte)
394d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
395b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    const ClientInfo &info = mClientInfos.valueFor(sessionID);
396d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
397d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString body = StringPrintf(
398d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_video_formats: "
399d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "30 00 02 02 00000040 00000000 00000000 00 0000 0000 00 none none\r\n"
400d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_audio_codecs: AAC 00000001 00\r\n"  // 2 ch AAC 48kHz
401d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_presentation_URL: rtsp://%s:%d/wfd1.0/streamid=0 none\r\n"
402d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        "wfd_client_rtp_ports: RTP/AVP/UDP;unicast 19000 0 mode=play\r\n",
403d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        info.mLocalIP.c_str(), info.mLocalPort);
404d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
405d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "SET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
406d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
407d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
408d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("Content-Type: text/parameters\r\n");
409d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(StringPrintf("Content-Length: %d\r\n", body.size()));
410d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("\r\n");
411d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(body);
412d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
413d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
414d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
415d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
416d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
417d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
418d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
419d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
420d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
421d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM4Response);
422d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
423d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
424d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
425d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
426d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
427d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
428d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::sendM5(int32_t sessionID) {
429d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString body = "wfd_trigger_method: SETUP\r\n";
430d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
431d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString request = "SET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
432d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
433d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
434d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("Content-Type: text/parameters\r\n");
435d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(StringPrintf("Content-Length: %d\r\n", body.size()));
436d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append("\r\n");
437d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    request.append(body);
438d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
439d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err =
440d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
441d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
442d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
443d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return err;
444d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
445d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
446d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    registerResponseHandler(
447d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM5Response);
448d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
449d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ++mNextCSeq;
450d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
451d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
452d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
453d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
454b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huberstatus_t WifiDisplaySource::sendM16(int32_t sessionID) {
455b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    AString request = "GET_PARAMETER rtsp://localhost/wfd1.0 RTSP/1.0\r\n";
456b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    AppendCommonResponse(&request, mNextCSeq);
457b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
458b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    const ClientInfo &info = mClientInfos.valueFor(sessionID);
459b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    request.append(StringPrintf("Session: %d\r\n", info.mPlaybackSessionID));
460a438123bd96c7faf145683876702387efe5628d9Andreas Huber    request.append("\r\n");  // Empty body
461b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
462b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    status_t err =
463b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        mNetSession->sendRequest(sessionID, request.c_str(), request.size());
464b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
465b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    if (err != OK) {
466b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        return err;
467b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    }
468b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
469b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    registerResponseHandler(
470b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            sessionID, mNextCSeq, &WifiDisplaySource::onReceiveM16Response);
471b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
472b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    ++mNextCSeq;
473b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
474b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    return OK;
475b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber}
476b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
477d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM1Response(
478d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
479d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
480d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
481d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
482d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
483d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
484d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
485d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
486d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
487d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
488d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
489d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
490d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
491d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM3Response(
492d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
493d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
494d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
495d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
496d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
497d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
498d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
499d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
500d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
501d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
502d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return sendM4(sessionID);
503d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
504d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
505d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM4Response(
506d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
507d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
508d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
509d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
510d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
511d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
512d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
513d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
514d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
515d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
516d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return sendM5(sessionID);
517d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
518d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
519d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberstatus_t WifiDisplaySource::onReceiveM5Response(
520d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
521d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t statusCode;
522d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!msg->getStatusCode(&statusCode)) {
523d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_MALFORMED;
524d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
525d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
526d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (statusCode != 200) {
527d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return ERROR_UNSUPPORTED;
528d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
529d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
530d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return OK;
531d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
532d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
533b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huberstatus_t WifiDisplaySource::onReceiveM16Response(
534b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        int32_t sessionID, const sp<ParsedMessage> &msg) {
535b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    // If only the response was required to include a "Session:" header...
536b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
537b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    const ClientInfo &info = mClientInfos.valueFor(sessionID);
538b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
539b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    ssize_t index = mPlaybackSessions.indexOfKey(info.mPlaybackSessionID);
540b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    if (index >= 0) {
541b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        mPlaybackSessions.valueAt(index)->updateLiveness();
542b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
543b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        scheduleKeepAlive(sessionID);
544b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    }
545b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
546b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    return OK;
547b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber}
548b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
549d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::scheduleReaper() {
550d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (mReaperPending) {
551d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
552d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
553d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
554d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    mReaperPending = true;
555d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    (new AMessage(kWhatReapDeadClients, id()))->post(kReaperIntervalUs);
556d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
557d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
558b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Hubervoid WifiDisplaySource::scheduleKeepAlive(int32_t sessionID) {
559b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    // We need to send updates at least 5 secs before the timeout is set to
560b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    // expire, make sure the timeout is greater than 5 secs to begin with.
561b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    CHECK_GT(kPlaybackSessionTimeoutUs, 5000000ll);
562b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
563b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    sp<AMessage> msg = new AMessage(kWhatKeepAlive, id());
564b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    msg->setInt32("sessionID", sessionID);
565b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    msg->post(kPlaybackSessionTimeoutUs - 5000000ll);
566b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber}
567b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
568d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onReceiveClientData(const sp<AMessage> &msg) {
569d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t sessionID;
570d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK(msg->findInt32("sessionID", &sessionID));
571d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
572d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<RefBase> obj;
573d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK(msg->findObject("data", &obj));
574d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
575d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<ParsedMessage> data =
576d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        static_cast<ParsedMessage *>(obj.get());
577d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
578d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ALOGV("session %d received '%s'",
579d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber          sessionID, data->debugString().c_str());
580d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
581d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString method;
582d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString uri;
583d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    data->getRequestField(0, &method);
584d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
585d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t cseq;
586d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!data->findInt32("cseq", &cseq)) {
587d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", -1 /* cseq */);
588d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
589d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
590d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
591d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (method.startsWith("RTSP/")) {
592d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        // This is a response.
593d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
594d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        ResponseID id;
595d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        id.mSessionID = sessionID;
596d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        id.mCSeq = cseq;
597d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
598d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        ssize_t index = mResponseHandlers.indexOfKey(id);
599d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
600d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (index < 0) {
601d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            ALOGW("Received unsolicited server response, cseq %d", cseq);
602d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            return;
603d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
604d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
605d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        HandleRTSPResponseFunc func = mResponseHandlers.valueAt(index);
606d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        mResponseHandlers.removeItemsAt(index);
607d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
608d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        status_t err = (this->*func)(sessionID, data);
609d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
610d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (err != OK) {
611d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            ALOGW("Response handler for session %d, cseq %d returned "
612d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                  "err %d (%s)",
613d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                  sessionID, cseq, err, strerror(-err));
614d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
615d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else {
616d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        AString version;
617d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        data->getRequestField(2, &version);
618d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (!(version == AString("RTSP/1.0"))) {
619d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "505 RTSP Version not supported", cseq);
620d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            return;
621d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
622d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
623d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (method == "DESCRIBE") {
624d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onDescribeRequest(sessionID, cseq, data);
625d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (method == "OPTIONS") {
626d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onOptionsRequest(sessionID, cseq, data);
627d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (method == "SETUP") {
628d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onSetupRequest(sessionID, cseq, data);
629d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (method == "PLAY") {
630d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onPlayRequest(sessionID, cseq, data);
631d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (method == "PAUSE") {
632d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onPauseRequest(sessionID, cseq, data);
633d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (method == "TEARDOWN") {
634d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onTeardownRequest(sessionID, cseq, data);
635d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (method == "GET_PARAMETER") {
636d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onGetParameterRequest(sessionID, cseq, data);
637d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (method == "SET_PARAMETER") {
638d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            onSetParameterRequest(sessionID, cseq, data);
639d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else {
640d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "405 Method Not Allowed", cseq);
641d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
642d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
643d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
644d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
645d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onDescribeRequest(
646d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
647d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
648d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
649d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int64_t nowUs = ALooper::GetNowUs();
650d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
651d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString sdp;
652d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sdp.append("v=0\r\n");
653d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
654d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sdp.append(StringPrintf(
655d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                "o=- %lld %lld IN IP4 0.0.0.0\r\n", nowUs, nowUs));
656d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
657d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sdp.append(
658d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "o=- 0 0 IN IP4 127.0.0.0\r\n"
659d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "s=Sample\r\n"
660d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "c=IN IP4 0.0.0.0\r\n"
661d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "b=AS:502\r\n"
662d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "t=0 0\r\n"
663d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "a=control:*\r\n"
664d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "a=range:npt=now-\r\n"
665d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "m=video 0 RTP/AVP 33\r\n"
666d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "a=rtpmap:33 MP2T/90000\r\n"
667d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "a=control:\r\n");
668d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
669d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
670d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq);
671d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
672d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("Content-Type: application/sdp\r\n");
673d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
674d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    // response.append("Content-Base: rtsp://0.0.0.0:7236\r\n");
675d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append(StringPrintf("Content-Length: %d\r\n", sdp.size()));
676d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
677d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append(sdp);
678d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
679d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
680d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
681d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
682d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
683d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onOptionsRequest(
684d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
685d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
686d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
687d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
688d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
689d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
690d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
691d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession != NULL) {
692d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        playbackSession->updateLiveness();
693d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
694d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
695d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
696d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq);
697d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
698d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append(
699d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "Public: org.wfa.wfd1.0, DESCRIBE, SETUP, TEARDOWN, PLAY, PAUSE, "
700d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            "GET_PARAMETER, SET_PARAMETER\r\n");
701d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
702d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
703d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
704d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
705d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
706d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
707d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    err = sendM3(sessionID);
708d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
709d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
710d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
711d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onSetupRequest(
712d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
713d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
714d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
715b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    ClientInfo *info = &mClientInfos.editValueFor(sessionID);
716b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    if (info->mPlaybackSessionID != -1) {
717b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        // We only support a single playback session per client.
718b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        // This is due to the reversed keep-alive design in the wfd specs...
719b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", cseq);
720b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        return;
721b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    }
722b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
723d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString transport;
724d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!data->findString("transport", &transport)) {
725d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", cseq);
726d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
727d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
728d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
729d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    bool useInterleavedTCP = false;
730d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
731d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int clientRtp, clientRtcp;
732d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (transport.startsWith("RTP/AVP/TCP;")) {
733d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        AString interleaved;
734d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (!ParsedMessage::GetAttribute(
735d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    transport.c_str(), "interleaved", &interleaved)
736d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                || sscanf(interleaved.c_str(), "%d-%d",
737d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          &clientRtp, &clientRtcp) != 2) {
738d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "400 Bad Request", cseq);
739d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            return;
740d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
741d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
742d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        useInterleavedTCP = true;
743d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else if (transport.startsWith("RTP/AVP;unicast;")
744d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            || transport.startsWith("RTP/AVP/UDP;unicast;")) {
745d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        bool badRequest = false;
746d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
747d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        AString clientPort;
748d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (!ParsedMessage::GetAttribute(
749d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    transport.c_str(), "client_port", &clientPort)) {
750d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            badRequest = true;
751d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (sscanf(clientPort.c_str(), "%d-%d",
752d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                          &clientRtp, &clientRtcp) == 2) {
753d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else if (sscanf(clientPort.c_str(), "%d", &clientRtp) == 1) {
754d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            // No RTCP.
755d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            clientRtcp = -1;
756d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else {
757d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            badRequest = true;
758d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
759d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
760d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (badRequest) {
761d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "400 Bad Request", cseq);
762d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            return;
763d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
764d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#if 1
765a438123bd96c7faf145683876702387efe5628d9Andreas Huber    // The older LG dongles doesn't specify client_port=xxx apparently.
766d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else if (transport == "RTP/AVP/UDP;unicast") {
767d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        clientRtp = 19000;
768d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        clientRtcp = clientRtp + 1;
769d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#endif
770d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else {
771d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "461 Unsupported Transport", cseq);
772d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
773d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
774d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
775d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID = makeUniquePlaybackSessionID();
776d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
777d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<AMessage> notify = new AMessage(kWhatPlaybackSessionNotify, id());
778d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    notify->setInt32("playbackSessionID", playbackSessionID);
779d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    notify->setInt32("sessionID", sessionID);
780d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
781d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
7820b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber        new PlaybackSession(
7830b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber                mNetSession, notify, mClient == NULL /* legacyMode */);
784d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
785d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    looper()->registerHandler(playbackSession);
786d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
787d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString uri;
788d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    data->getRequestField(1, &uri);
789d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
790d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (strncasecmp("rtsp://", uri.c_str(), 7)) {
791d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "400 Bad Request", cseq);
792d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
793d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
794d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
795d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!(uri.startsWith("rtsp://") && uri.endsWith("/wfd1.0/streamid=0"))) {
796d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "404 Not found", cseq);
797d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
798d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
799d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
800d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = playbackSession->init(
801b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            info->mRemoteIP.c_str(),
802d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            clientRtp,
803d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            clientRtcp,
804d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            useInterleavedTCP);
805d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
806d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (err != OK) {
807d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        looper()->unregisterHandler(playbackSession->id());
808d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        playbackSession.clear();
809d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
810d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
811d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    switch (err) {
812d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case OK:
813d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            break;
814d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        case -ENOENT:
815d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "404 Not Found", cseq);
816d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            return;
817d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        default:
818d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            sendErrorResponse(sessionID, "403 Forbidden", cseq);
819d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            return;
820d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
821d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
822d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    mPlaybackSessions.add(playbackSessionID, playbackSession);
823d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
824b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    info->mPlaybackSessionID = playbackSessionID;
825b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
826d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
827d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
828d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
829d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (useInterleavedTCP) {
830d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        response.append(
831d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                StringPrintf(
832d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    "Transport: RTP/AVP/TCP;interleaved=%d-%d;",
833d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    clientRtp, clientRtcp));
834d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    } else {
835d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t serverRtp = playbackSession->getRTPPort();
836d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
837d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        if (clientRtcp >= 0) {
838d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response.append(
839d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    StringPrintf(
840d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        "Transport: RTP/AVP;unicast;client_port=%d-%d;"
841d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        "server_port=%d-%d\r\n",
842d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        clientRtp, clientRtcp, serverRtp, serverRtp + 1));
843d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        } else {
844d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            response.append(
845d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    StringPrintf(
846d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        "Transport: RTP/AVP;unicast;client_port=%d;"
847d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        "server_port=%d\r\n",
848d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                        clientRtp, serverRtp));
849d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
850d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
851d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
852d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
853d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
854d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    err = mNetSession->sendRequest(sessionID, response.c_str());
855d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
856d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
857d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    scheduleReaper();
858b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber    scheduleKeepAlive(sessionID);
859d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
860d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
861d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onPlayRequest(
862d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
863d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
864d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
865d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
866d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
867d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
868d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
869d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
870d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
871d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
872d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
873d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
874d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = playbackSession->play();
875d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
876d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
877d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
878d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
879d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("Range: npt=now-\r\n");
880d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
881d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
882d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    err = mNetSession->sendRequest(sessionID, response.c_str());
883d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
8840b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber
8850b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber    if (mClient != NULL) {
8860b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber        mClient->onDisplayConnected(
8870b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber                playbackSession->getSurfaceTexture(),
8880b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber                playbackSession->width(),
8890b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber                playbackSession->height(),
8900b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber                0 /* flags */);
8910b73d4730202fcad53aefc4314a06e7b95f442f0Andreas Huber    }
892d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
893d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
894d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onPauseRequest(
895d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
896d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
897d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
898d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
899d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
900d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
901d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
902d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
903d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
904d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
905d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
906d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
907d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = playbackSession->pause();
908d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
909d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
910d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
911d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
912d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
913d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
914d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    err = mNetSession->sendRequest(sessionID, response.c_str());
915d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
916d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
917d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
918d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onTeardownRequest(
919d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
920d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
921d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
922d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
923d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
924d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
925d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
926d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
927d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
928d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
929d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
930d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
931d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    looper()->unregisterHandler(playbackSession->id());
932d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    mPlaybackSessions.removeItem(playbackSessionID);
933d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
934d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
935d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
936d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
937d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
938d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
939d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
940d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
941d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
942d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onGetParameterRequest(
943d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
944d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
945d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
946d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
947d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
948d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
949d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
950d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
951d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
952d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
953d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
954d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
955d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    playbackSession->updateLiveness();
956d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
957d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
958d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
959d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
960d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
961d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
962d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
963d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
964d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
965d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::onSetParameterRequest(
966d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
967d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq,
968d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data) {
969d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    int32_t playbackSessionID;
970d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    sp<PlaybackSession> playbackSession =
971d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        findPlaybackSession(data, &playbackSessionID);
972d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
973a438123bd96c7faf145683876702387efe5628d9Andreas Huber#if 1
974a438123bd96c7faf145683876702387efe5628d9Andreas Huber    // XXX the older dongles do not include a "Session:" header in this request.
975a438123bd96c7faf145683876702387efe5628d9Andreas Huber    if (playbackSession == NULL) {
976a438123bd96c7faf145683876702387efe5628d9Andreas Huber        CHECK_EQ(mPlaybackSessions.size(), 1u);
977a438123bd96c7faf145683876702387efe5628d9Andreas Huber        playbackSessionID = mPlaybackSessions.keyAt(0);
978a438123bd96c7faf145683876702387efe5628d9Andreas Huber        playbackSession = mPlaybackSessions.valueAt(0);
979a438123bd96c7faf145683876702387efe5628d9Andreas Huber    }
980a438123bd96c7faf145683876702387efe5628d9Andreas Huber#else
981d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSession == NULL) {
982d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        sendErrorResponse(sessionID, "454 Session Not Found", cseq);
983d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return;
984d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
985d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber#endif
986d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
987d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    playbackSession->updateLiveness();
988d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
989d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response = "RTSP/1.0 200 OK\r\n";
990d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq, playbackSessionID);
991d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
992d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
993d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
994d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
995d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
996d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
997d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber// static
998d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::AppendCommonResponse(
999d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        AString *response, int32_t cseq, int32_t playbackSessionID) {
1000d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    time_t now = time(NULL);
1001d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    struct tm *now2 = gmtime(&now);
1002d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    char buf[128];
1003d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    strftime(buf, sizeof(buf), "%a, %d %b %Y %H:%M:%S %z", now2);
1004d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1005d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append("Date: ");
1006d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append(buf);
1007d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append("\r\n");
1008d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1009d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response->append("Server: Mine/1.0\r\n");
1010d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1011d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (cseq >= 0) {
1012d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        response->append(StringPrintf("CSeq: %d\r\n", cseq));
1013d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1014d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1015d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (playbackSessionID >= 0ll) {
1016d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        response->append(
1017d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                StringPrintf(
1018d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    "Session: %d;timeout=%lld\r\n",
1019d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                    playbackSessionID, kPlaybackSessionTimeoutSecs));
1020d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1021d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1022d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1023d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubervoid WifiDisplaySource::sendErrorResponse(
1024d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t sessionID,
1025d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const char *errorDetail,
1026d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t cseq) {
1027d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AString response;
1028d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("RTSP/1.0 ");
1029d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append(errorDetail);
1030d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1031d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1032d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    AppendCommonResponse(&response, cseq);
1033d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1034d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    response.append("\r\n");
1035d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1036d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    status_t err = mNetSession->sendRequest(sessionID, response.c_str());
1037d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    CHECK_EQ(err, (status_t)OK);
1038d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1039d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1040d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huberint32_t WifiDisplaySource::makeUniquePlaybackSessionID() const {
1041d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    for (;;) {
1042d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        int32_t playbackSessionID = rand();
1043d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1044b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        if (playbackSessionID == -1) {
1045b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            // reserved.
1046b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber            continue;
1047b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber        }
1048b6777017a68ed473d61cc9d6e77c34fd5cd301ccAndreas Huber
1049d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        for (size_t i = 0; i < mPlaybackSessions.size(); ++i) {
1050d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            if (mPlaybackSessions.keyAt(i) == playbackSessionID) {
1051d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber                continue;
1052d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber            }
1053d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        }
1054d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1055d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return playbackSessionID;
1056d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1057d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1058d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1059d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Hubersp<WifiDisplaySource::PlaybackSession> WifiDisplaySource::findPlaybackSession(
1060d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        const sp<ParsedMessage> &data, int32_t *playbackSessionID) const {
1061d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (!data->findInt32("session", playbackSessionID)) {
1062d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        *playbackSessionID = 0;
1063d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return NULL;
1064d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1065d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1066d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    ssize_t index = mPlaybackSessions.indexOfKey(*playbackSessionID);
1067d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    if (index < 0) {
1068d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber        return NULL;
1069d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    }
1070d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1071d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber    return mPlaybackSessions.valueAt(index);
1072d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}
1073d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1074d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber}  // namespace android
1075d7bee3a9d2ad76d073d91f0ee36d5ac5f9df480cAndreas Huber
1076