source: roaraudio/libroar/basic.c @ 522:831c6669c72a

Last change on this file since 522:831c6669c72a was 522:831c6669c72a, checked in by phi, 16 years ago

try also DECnet if server is NULL

File size: 6.4 KB
Line 
1//basic.c:
2
3#include "libroar.h"
4
5int roar_connect_raw (char * server) {
6 char user_sock[80];
7 char * roar_server;
8 int i;
9 int port = 0;
10 int fh = -1;
11 int is_decnet = 0;
12 char * obj = NULL;
13
14 if ( server == NULL && (roar_server = getenv("ROAR_SERVER")) != NULL )
15  server = roar_server;
16
17 if ( server == NULL && (i = readlink("/etc/roarserver", user_sock, 79)) != -1 ) {
18   user_sock[i] = 0;
19   server = user_sock;
20 }
21
22 if ( server == NULL || *server == 0 ) {
23  /* connect via defaults */
24
25
26  snprintf(user_sock, 79, "%s/%s", getenv("HOME"), ROAR_DEFAULT_SOCK_USER);
27
28  if ( (fh = roar_socket_connect(user_sock, 0)) != -1 )
29   return fh;
30
31  if ( (fh = roar_socket_connect(ROAR_DEFAULT_SOCK_GLOBAL, 0)) != -1 )
32   return fh;
33
34  if ( (fh = roar_socket_connect(ROAR_DEFAULT_HOST, ROAR_DEFAULT_PORT)) != -1 )
35   return fh;
36
37  if ( roar_socket_get_local_nodename() ) {
38   snprintf(user_sock, 79, "%s::%s", roar_socket_get_local_nodename(), ROAR_DEFAULT_OBJECT);
39   return roar_socket_connect(user_sock, ROAR_DEFAULT_NUM);
40  }
41
42 } else {
43  /* connect via (char*)server */
44  // find a port:
45  if ( *server != '/' ) { // don't test AF_UNIX sockets for ports
46   for (i = 0; server[i] != 0; i++) {
47    if ( server[i] == ':' ) {
48     if ( server[i+1] == ':' ) { // DECnet, leave unchanged
49      is_decnet = 1;
50      obj = &server[i+2];
51      break;
52     }
53
54     port = atoi(server+i+1);
55     server[i] = 0;
56     break;
57    }
58   }
59  }
60
61  if ( is_decnet ) {
62    *user_sock = 0;
63   if ( *server == ':' ) {
64    if ( roar_socket_get_local_nodename() )
65     strcat(user_sock, roar_socket_get_local_nodename());
66   }
67
68   strcat(user_sock, server);
69   server = user_sock;
70   if ( *obj == 0 ) {
71    strcat(user_sock, ROAR_DEFAULT_OBJECT);
72   }
73  }
74
75  if ( port || is_decnet ) {
76   fh = roar_socket_connect(server, port);
77   // restore the original string
78   server[i] = ':';
79  } else {
80   fh = roar_socket_connect(server, ROAR_DEFAULT_PORT);
81  }
82 }
83
84 ROAR_DBG("roar_connect_raw(*) = %i", fh);
85
86 return fh;
87}
88
89int roar_connect    (struct roar_connection * con, char * server) {
90 con->fh = roar_connect_raw(server);
91
92 if ( con->fh == -1 )
93  return -1;
94
95 return 0;
96}
97
98int roar_disconnect (struct roar_connection * con) {
99 struct roar_message m;
100
101 m.datalen = 0;
102 m.stream  = 0;
103 m.pos     = 0;
104 m.cmd     = ROAR_CMD_QUIT;
105
106 roar_req(con, &m, NULL);
107
108 close(con->fh);
109
110 con->fh = -1;
111
112 return 0;
113}
114
115int roar_identify   (struct roar_connection * con, char * name) {
116 struct roar_message mes;
117 pid_t pid;
118 int max_len;
119
120 ROAR_DBG("roar_identify(*): try to identify myself...");
121
122 mes.cmd    = ROAR_CMD_IDENTIFY;
123 mes.stream = 0;
124 mes.pos    = 0;
125
126 ROAR_DBG("roar_identify(*): name=%p", name);
127
128 if ( name == NULL )
129  name = "libroar client";
130
131 ROAR_DBG("roar_identify(*): name=%p", name);
132
133 max_len = strlen(name);
134 ROAR_DBG("roar_identify(*): strlen(name) = %i", max_len);
135
136 if ( max_len > (LIBROAR_BUFFER_MSGDATA - 5) )
137  max_len = LIBROAR_BUFFER_MSGDATA - 5;
138
139 mes.datalen = 5 + max_len;
140 mes.data[0] = 1;
141
142 pid = getpid();
143 *(uint32_t*)(mes.data+1) = ROAR_HOST2NET32(pid);
144 ROAR_DBG("roar_identify(*): pid = %i", pid);
145
146 strncpy(mes.data+5, name, max_len);
147
148 return roar_req(con, &mes, NULL);
149}
150
151#define _ROAR_MESS_BUF_LEN (1 /* version */ + 1 /* cmd */ + 2 /* stream */ + 4 /* pos */ + 2 /* datalen */)
152int roar_send_message (struct roar_connection * con, struct roar_message * mes, char * data) {
153 char buf[_ROAR_MESS_BUF_LEN];
154
155 ROAR_DBG("roar_send_message(*): try to send an request...");
156
157 buf[0] = _ROAR_MESSAGE_VERSION;
158 buf[1] = (unsigned char) mes->cmd;
159 *(uint16_t*)(buf+2) = ROAR_HOST2NET16(mes->stream);
160 *(uint32_t*)(buf+4) = ROAR_HOST2NET32(mes->pos);
161 *(uint16_t*)(buf+8) = ROAR_HOST2NET16(mes->datalen);
162
163 if ( write(con->fh, buf, _ROAR_MESS_BUF_LEN) != _ROAR_MESS_BUF_LEN )
164  return -1;
165
166 if ( mes->datalen != 0 )
167  if ( write(con->fh, data == NULL ? mes->data : data, mes->datalen) != mes->datalen )
168   return -1;
169
170 ROAR_DBG("roar_send_message(*) = 0");
171 return 0;
172}
173
174int roar_recv_message (struct roar_connection * con, struct roar_message * mes, char ** data) {
175 char buf[_ROAR_MESS_BUF_LEN];
176
177 ROAR_DBG("roar_recv_message(*): try to get a response form the server...");
178
179 if ( data )
180  *data = NULL;
181
182 if ( read(con->fh, buf, _ROAR_MESS_BUF_LEN) != _ROAR_MESS_BUF_LEN )
183  return -1;
184
185 ROAR_DBG("roar_recv_message(*): Got a header");
186
187 if ( buf[0] != _ROAR_MESSAGE_VERSION )
188  return -1;
189
190 mes->cmd     = (unsigned char)buf[1];
191 mes->stream  = ROAR_NET2HOST16(*(uint16_t*)(buf+2));
192 mes->pos     = ROAR_NET2HOST32(*(uint32_t*)(buf+4));
193 mes->datalen = ROAR_NET2HOST16(*(uint16_t*)(buf+8));
194
195 ROAR_DBG("roar_recv_message(*): command=%i(%s)", mes->cmd,
196           mes->cmd == ROAR_CMD_OK ? "OK" : (mes->cmd == ROAR_CMD_ERROR ? "ERROR" : "UNKNOWN"));
197
198 if ( mes->datalen == 0 ) {
199  ROAR_DBG("roar_recv_message(*): no data in this pkg");
200  ROAR_DBG("roar_recv_message(*) = 0");
201  return 0;
202 }
203
204 if ( mes->datalen <= LIBROAR_BUFFER_MSGDATA ) {
205  if ( read(con->fh, mes->data, mes->datalen) == mes->datalen ) {
206   ROAR_DBG("roar_recv_message(*): Got data!");
207   ROAR_DBG("roar_recv_message(*) = 0");
208   return 0;
209  }
210  return -1;
211 } else {
212  if ( data == NULL )
213   return -1;
214
215  if ( (*data = malloc(mes->datalen)) == NULL )
216   return -1;
217
218  if ( mes->datalen == 0 )
219   return 0;
220
221  if ( read(con->fh, *data, mes->datalen) == mes->datalen ) {
222   ROAR_DBG("roar_recv_message(*): Got data!");
223   ROAR_DBG("roar_recv_message(*) = 0");
224   return 0;
225  }
226  return -1;
227 }
228
229 return -1;
230}
231
232int roar_req (struct roar_connection * con, struct roar_message * mes, char ** data) {
233 if ( roar_send_message(con, mes, data ? *data : NULL) != 0 )
234  return -1;
235
236 if ( data )
237  free(*data);
238
239 return roar_recv_message(con, mes, data);
240}
241
242int roar_debug_message_print (struct roar_message * mes) {
243 if ( mes == NULL )
244  return -1;
245
246 ROAR_DBG("roar_debug_message_print(*): Command: %i", mes->cmd);
247 ROAR_DBG("roar_debug_message_print(*): Stream : %u", mes->stream);
248 ROAR_DBG("roar_debug_message_print(*): Pos    : %u", mes->pos);
249 ROAR_DBG("roar_debug_message_print(*): Datalen: %i", mes->datalen);
250
251 ROAR_DBG("roar_debug_message_print(*) = 0");
252 return 0;
253}
254
255int roar_debug_audio_info_print (struct roar_audio_info * info) {
256 if ( info == NULL )
257  return -1;
258
259 ROAR_DBG("roar_debug_audio_info_print(*): Rate    : %i", info->rate);
260 ROAR_DBG("roar_debug_audio_info_print(*): Channels: %i", info->channels);
261 ROAR_DBG("roar_debug_audio_info_print(*): Bits    : %i", info->bits);
262 ROAR_DBG("roar_debug_audio_info_print(*): Codec   : %i", info->codec);
263
264 ROAR_DBG("roar_debug_audio_info_print(*) = 0");
265 return 0;
266}
267
268//ll
Note: See TracBrowser for help on using the repository browser.