source: roaraudio/roard/network.c @ 3684:69134bcda4c2

Last change on this file since 3684:69134bcda4c2 was 3684:69134bcda4c2, checked in by phi, 14 years ago

added most of the rsound interface

File size: 4.8 KB
Line 
1//network.c:
2
3/*
4 *      Copyright (C) Philipp 'ph3-der-loewe' Schafft - 2008-2010
5 *
6 *  This file is part of roard a part of RoarAudio,
7 *  a cross-platform sound system for both, home and professional use.
8 *  See README for details.
9 *
10 *  This file is free software; you can redistribute it and/or modify
11 *  it under the terms of the GNU General Public License version 3
12 *  as published by the Free Software Foundation.
13 *
14 *  RoarAudio is distributed in the hope that it will be useful,
15 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
16 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 *  GNU General Public License for more details.
18 *
19 *  You should have received a copy of the GNU General Public License
20 *  along with this software; see the file COPYING.  If not, write to
21 *  the Free Software Foundation, 51 Franklin Street, Fifth Floor,
22 *  Boston, MA 02110-1301, USA.
23 *
24 */
25
26#include "roard.h"
27
28#ifdef ROAR_SUPPORT_LISTEN
29
30#ifdef ROAR_BROKEN_PEERCRED
31#undef SO_PEERCRED
32#endif
33
34#ifdef ROAR_HAVE_SELECT
35#define _CAN_OPERATE
36#endif
37
38int net_check_listen  (void) {
39#ifdef _CAN_OPERATE
40 int r;
41 fd_set sl;
42 struct timeval tv;
43 int i;
44 int max_fh = -1;
45
46 FD_ZERO(&sl);
47
48 for (i = 0; i < ROAR_MAX_LISTEN_SOCKETS; i++) {
49  if ( g_listen[i].socket != -1 ) {
50   if ( g_listen[i].socket > max_fh )
51    max_fh = g_listen[i].socket;
52
53   FD_SET(g_listen[i].socket, &sl);
54  }
55 }
56
57 if ( max_fh == -1 )
58  return 0;
59
60 tv.tv_sec  = 0;
61 tv.tv_usec = 1;
62
63 if ((r = select(max_fh + 1, &sl, NULL, NULL, &tv)) > 0) {
64  ROAR_DBG("net_check_listen(void): We have a connection!");
65  for (i = 0; i < ROAR_MAX_LISTEN_SOCKETS; i++) {
66   if ( g_listen[i].socket != -1 ) {
67    if ( FD_ISSET(g_listen[i].socket, &sl) ) {
68     if ( net_get_new_client(&(g_listen[i])) == -1 )
69      return -1;
70    }
71   }
72  }
73 }
74
75 return r;
76#else
77 return -1;
78#endif
79}
80
81#ifdef _CAN_OPERATE
82int net_get_new_client (struct roard_listen * lsock) {
83 int fh;
84 int client;
85 struct roar_client * c;
86#ifdef SO_PEERCRED
87 struct ucred cred;
88 socklen_t cred_len = sizeof(cred);
89#endif
90 struct roar_vio_calls    vio;
91 struct sockaddr_storage  addr;
92 socklen_t                addrlen = sizeof(addr);
93
94 fh = accept(lsock->socket, (struct sockaddr*)&addr, &addrlen);
95
96 ROAR_DBG("net_get_new_client(void): fh = %i", fh);
97
98#ifndef ROAR_WITHOUT_DCOMP_EMUL_RSOUND
99 if ( lsock->proto == ROAR_PROTO_RSOUND ) {
100  client = emul_rsound_on_connect(fh, lsock);
101  switch (client) {
102   case -1: return -1; break;
103   case -2: return  0; break;
104   default: // TODO: write error handling
105     clients_get(client, &c);
106     fh = c->fh;
107    break;
108  }
109 } else {
110#endif
111
112 client = clients_new();
113
114 if ( clients_set_fh(client, fh) == -1 ) {
115  ROAR_ERR("net_get_new_client(void): Can not set client's fh");
116
117  clients_delete(client);
118  close(fh);
119
120  ROAR_DBG("net_get_new_client(void) = -1");
121  return -1;
122 }
123#ifndef ROAR_WITHOUT_DCOMP_EMUL_RSOUND
124 }
125#endif
126
127 if ( clients_get(client, &c) != -1 ) {
128#ifdef SO_PEERCRED
129  if (getsockopt(fh, SOL_SOCKET, SO_PEERCRED, &cred, &cred_len) != -1) {
130   if ( cred.pid != 0 ) {
131    c->pid = cred.pid;
132    c->uid = cred.uid;
133    c->gid = cred.gid;
134   }
135  } else {
136   ROAR_DBG("req_on_identify(): Can't get creds via SO_PEERCRED: %s", strerror(errno));
137  }
138#elif defined(ROAR_HAVE_GETPEEREID)
139  if (getpeereid(fh, &(c->uid), &(c->gid)) == -1) {
140   ROAR_DBG("req_on_identify(): Can't get creds via getpeereid(): %s", strerror(errno));
141  }
142#endif
143
144  if ( roar_nnode_free(&(c->nnode)) == -1 )
145   return -1;
146
147  if ( roar_nnode_new_from_sockaddr(&(c->nnode), (struct sockaddr*)&addr, addrlen) == -1 )
148   return -1;
149 }
150
151 ROAR_DBG("net_get_new_client(*): proto=0x%.4x", lsock->proto);
152
153 if ( clients_set_proto(client, lsock->proto) == -1 ) {
154  ROAR_WARN("net_get_new_client(*): Setting proto(0x%.4x) of client %i failed.", lsock->proto, client);
155  return -1;
156 }
157
158 switch (lsock->proto) {
159  case ROAR_PROTO_ROARAUDIO:
160    // nothing needed to be done here
161   break;
162#ifndef ROAR_WITHOUT_DCOMP_EMUL_ESD
163#ifdef ROAR_HAVE_H_ESD
164  case ROAR_PROTO_ESOUND:
165    ROAR_DBG("net_get_new_client(*): execing ESD CONNECT command");
166
167    if ( roar_vio_open_fh_socket(&vio, fh) == -1 )
168     return -1;
169
170    ROAR_DBG("net_get_new_client(*): creating VIO OK");
171
172    if ( emul_esd_exec_command(client, ESD_PROTO_CONNECT, &vio) == -1 )
173     return -1;
174
175    ROAR_DBG("net_get_new_client(*): CONNECT execed sucessfully");
176   break;
177#endif
178#endif
179#ifndef ROAR_WITHOUT_DCOMP_EMUL_SIMPLE
180  case ROAR_PROTO_SIMPLE:
181    if ( emul_simple_on_connect(client, lsock) == -1 )
182     return -1;
183   break;
184#endif
185#ifndef ROAR_WITHOUT_DCOMP_EMUL_RSOUND
186  case ROAR_PROTO_RSOUND: // nothing to do here.
187   break;
188#endif
189  default:
190    // OS independiend code to close the socket:
191    if ( roar_vio_open_fh_socket(&vio, fh) == -1 )
192     return -1;
193    roar_vio_close(&vio);
194    return -1;
195   break;
196 }
197
198// close(fh);
199
200 return 0;
201}
202#endif
203
204#endif
205
206//ll
Note: See TracBrowser for help on using the repository browser.