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