1 /*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 #include "RenderServer.h"
17 #include "TcpStream.h"
18 #ifdef _WIN32
19 #include "Win32PipeStream.h"
20 #else
21 #include "UnixStream.h"
22 #endif
23 #include "RenderThread.h"
24 #include "FrameBuffer.h"
25 #include <set>
26 
27 typedef std::set<RenderThread *> RenderThreadsSet;
28 
RenderServer()29 RenderServer::RenderServer() :
30     m_listenSock(NULL),
31     m_exiting(false)
32 {
33 }
34 
~RenderServer()35 RenderServer::~RenderServer()
36 {
37     delete m_listenSock;
38 }
39 
40 
41 extern "C" int gRendererStreamMode;
42 
create(char * addr,size_t addrLen)43 RenderServer *RenderServer::create(char* addr, size_t addrLen)
44 {
45     RenderServer *server = new RenderServer();
46     if (!server) {
47         return NULL;
48     }
49 
50     if (gRendererStreamMode == STREAM_MODE_TCP) {
51         server->m_listenSock = new TcpStream();
52     } else {
53 #ifdef _WIN32
54         server->m_listenSock = new Win32PipeStream();
55 #else
56         server->m_listenSock = new UnixStream();
57 #endif
58     }
59 
60     char addrstr[SocketStream::MAX_ADDRSTR_LEN];
61     if (server->m_listenSock->listen(addrstr) < 0) {
62         ERR("RenderServer::create failed to listen\n");
63         delete server;
64         return NULL;
65     }
66 
67     size_t len = strlen(addrstr) + 1;
68     if (len > addrLen) {
69         ERR("RenderServer address name too big for provided buffer: %zu > %zu\n",
70                 len, addrLen);
71         delete server;
72         return NULL;
73     }
74     memcpy(addr, addrstr, len);
75 
76     return server;
77 }
78 
Main()79 int RenderServer::Main()
80 {
81     RenderThreadsSet threads;
82 
83     while(1) {
84         SocketStream *stream = m_listenSock->accept();
85         if (!stream) {
86             fprintf(stderr,"Error accepting connection, aborting\n");
87             break;
88         }
89 
90         unsigned int clientFlags;
91         if (!stream->readFully(&clientFlags, sizeof(unsigned int))) {
92             fprintf(stderr,"Error reading clientFlags\n");
93             delete stream;
94             continue;
95         }
96 
97         DBG("RenderServer: Got new stream!\n");
98 
99         // check if we have been requested to exit while waiting on accept
100         if ((clientFlags & IOSTREAM_CLIENT_EXIT_SERVER) != 0) {
101             m_exiting = true;
102             break;
103         }
104 
105         RenderThread *rt = RenderThread::create(stream);
106         if (!rt) {
107             fprintf(stderr,"Failed to create RenderThread\n");
108             delete stream;
109             stream = NULL;
110         } else if (!rt->start()) {
111             fprintf(stderr,"Failed to start RenderThread\n");
112             delete rt;
113             rt = NULL;
114         }
115 
116         //
117         // remove from the threads list threads which are
118         // no longer running
119         //
120         for (RenderThreadsSet::iterator n,t = threads.begin();
121              t != threads.end();
122              t = n) {
123             // first find next iterator
124             n = t;
125             n++;
126 
127             // delete and erase the current iterator
128             // if thread is no longer running
129             if ((*t)->isFinished()) {
130                 delete (*t);
131                 threads.erase(t);
132             }
133         }
134 
135         // if the thread has been created and started, insert it to the list
136         if (rt) {
137             threads.insert(rt);
138             DBG("Started new RenderThread\n");
139         }
140     }
141 
142     //
143     // Wait for all threads to finish
144     //
145     for (RenderThreadsSet::iterator t = threads.begin();
146          t != threads.end();
147          t++) {
148         int exitStatus;
149         (*t)->wait(&exitStatus);
150         delete (*t);
151     }
152     threads.clear();
153 
154     //
155     // de-initialize the FrameBuffer object
156     //
157     FrameBuffer::finalize();
158     return 0;
159 }
160