e6f16a93239eb6f4ffcb19ff087171505262de05
[blender.git] / source / blender / blenkernel / intern / verse_session.c
1 /**
2  * $Id$
3  *
4  * ***** BEGIN GPL/BL DUAL LICENSE BLOCK *****
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License
8  * as published by the Free Software Foundation; either version 2
9  * of the License, or (at your option) any later version. The Blender
10  * Foundation also sells licenses for use in proprietary software under
11  * the Blender License.  See http://www.blender.org/BL/ for information
12  * about this.
13  *
14  * This program 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 program; if not, write to the Free Software Foundation,
21  * Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22  *
23  * Contributor(s): Jiri Hnidek.
24  *
25  * ***** END GPL/BL DUAL LICENSE BLOCK *****
26  */
27
28 #ifdef WITH_VERSE
29
30 #include <string.h>
31
32 #include "MEM_guardedalloc.h"
33
34 #include "DNA_mesh_types.h"     /* temp */
35 #include "DNA_listBase.h"
36 #include "DNA_screen_types.h"
37
38 #include "BLI_dynamiclist.h"
39 #include "BLI_blenlib.h"
40
41 #include "BIF_verse.h"
42
43 #include "BKE_global.h" 
44 #include "BKE_verse.h"
45
46 #include "verse.h"
47
48 struct ListBase session_list={NULL, NULL};
49
50 /* list of static function prototypes */
51 static void cb_connect_terminate(const char *address, const char *bye);
52 static void cb_connect_accept(void *user_data, uint32 avatar, void *address, void *connection, const uint8 *host_id);
53 static void set_all_callbacks(void);
54 static void free_verse_session_data(struct VerseSession *session);
55
56 /*
57  * callback function for connection terminated
58  */
59 static void cb_connect_terminate(const char *address, const char *bye)
60 {
61         VerseSession *session = (VerseSession*)current_verse_session();
62
63         if(!session) return;
64
65         /* remove session from list of session */
66         BLI_remlink(&session_list, session);
67         /* do post connect operations */
68         session->post_connect_terminated(session);
69         /* free session data */
70         free_verse_session_data(session);
71         /* free session */
72         MEM_freeN(session);
73 }
74
75 /*
76  * callback function for accepted connection to verse server
77  */
78 static void cb_connect_accept(
79                 void *user_data,
80                 uint32 avatar,
81                 void *address,
82                 void *connection,
83                 const uint8 *host_id)
84 {
85         struct VerseSession *session = (VerseSession*)current_verse_session();
86         uint32 i, mask=0;
87
88         if(!session) return;
89
90         session->flag |= VERSE_CONNECTED;
91         session->flag &= ~VERSE_CONNECTING;
92
93         printf("\tBlender was connected to verse server: %s\n", (char*)address);
94         printf("\tVerseSession->counter: %d\n", session->counter);
95
96         session->avatar = avatar;
97
98         session->post_connect_accept(session);
99
100         for(i = 0; i < V_NT_NUM_TYPES; i++)
101                 mask = mask | (1 << i);
102         verse_send_node_index_subscribe(mask);
103 }
104
105 /*
106  * set up all callbacks for sessions
107  */
108 void set_verse_session_callbacks(void)
109 {
110         /* connection */
111         verse_callback_set(verse_send_connect_accept, cb_connect_accept, NULL);
112         /* connection was terminated */
113         verse_callback_set(verse_send_connect_terminate, cb_connect_terminate, NULL);
114 }
115
116 /*
117  * set all callbacks used in Blender
118  */
119 static void set_all_callbacks(void)
120 {
121         /* set up all callbacks for sessions */
122         set_verse_session_callbacks();
123
124         /* set up callbacks for nodes */
125         set_node_callbacks();
126
127         /* set up all callbacks for object nodes */
128         set_object_callbacks();
129
130         /* set up all callbacks for geometry nodes */
131         set_geometry_callbacks();
132
133         /* set up all callbacks for bitmap nodes */
134         set_bitmap_callbacks();
135 }
136
137 /*
138  * this function sends and receive all packets for all sessions
139  */
140 void b_verse_update(void)
141 {
142         VerseSession *session, *next_session;
143
144         session = session_list.first;
145         while(session){
146                 next_session = session->next;
147                 verse_session_set(session->vsession);
148                 if((session->flag & VERSE_CONNECTED) || (session->flag & VERSE_CONNECTING)) {
149                         verse_callback_update(10);
150                         session->post_connect_update(session);
151                 }
152                 session = next_session;
153         }
154 }
155
156 /*
157  * returns VerseSession coresponding to vsession pointer
158  */
159 VerseSession *versesession_from_vsession(VSession *vsession)
160 {
161         struct VerseSession *session;
162
163         session = session_list.first;
164
165         while(session) {
166                 if(session->vsession==vsession) return session;
167                 session = session->next;
168         }
169         
170         return session;
171 }
172
173 /*
174  * returns pointer at current VerseSession
175  */
176 VerseSession *current_verse_session(void)
177 {
178         struct VerseSession *session;
179         VSession vsession = verse_session_get();
180
181         session = session_list.first;
182
183         while(session){
184                 if(session->vsession == vsession)
185                         return session;
186                 session = session->next;
187         }
188
189         printf("error: non-existing SESSION occured!\n");
190         return NULL;
191 }
192
193 /*
194  * free VerseSession
195  */
196 static void free_verse_session_data(VerseSession *session)
197 {
198         struct VNode *vnode;
199
200         /* free data of all nodes */
201         vnode = session->nodes.lb.first;
202         while(vnode){
203                 free_verse_node_data(vnode);
204                 vnode = vnode->next;
205         }
206
207         /* free data of nodes waiting in queue */
208         vnode = session->queue.first;
209         while(vnode){
210                 free_verse_node_data(vnode);
211                 vnode = vnode->next;
212         }
213
214         /* free all VerseNodes */
215         BLI_dlist_destroy(&(session->nodes));
216         /* free all VerseNodes waiting in queque */
217         BLI_freelistN(&(session->queue));
218
219         /* free name of verse host for this session */
220         MEM_freeN(session->address);
221 }
222
223 /*
224  * free VerseSession
225  */
226 void free_verse_session(VerseSession *session)
227 {
228         /* remove session from session list*/
229         BLI_remlink(&session_list, session);
230         /* do post terminated operations */
231         session->post_connect_terminated(session);
232         /* free session data (nodes, layers) */
233         free_verse_session_data(session);
234         /* free session */
235         MEM_freeN(session);
236 }
237
238 /*
239  * create new verse session and return coresponding data structure
240  */
241 VerseSession *create_verse_session(
242                 const char *name,
243                 const char *pass,
244                 const char *address,
245                 uint8 *expected_key)
246 {
247         struct VerseSession *session;
248         VSession *vsession;
249         
250         vsession = verse_send_connect(name, pass, address, expected_key);
251
252         if(!vsession) return NULL;
253
254         session = (VerseSession*)MEM_mallocN(sizeof(VerseSession), "VerseSession");
255
256         session->flag = VERSE_CONNECTING;
257
258         session->vsession = vsession;
259         session->avatar = -1;
260
261         session->address = (char*)MEM_mallocN(sizeof(char)*(strlen(address)+1),"session adress name");
262         strcpy(session->address, address);
263
264         session->connection = NULL;
265         session->host_id = NULL;
266         session->counter = 0;
267
268         /* initialize dynamic list of nodes and node queue */
269         BLI_dlist_init(&(session->nodes));
270         session->queue.first = session->queue.last = NULL;
271
272         /* set up all client dependent functions */
273         session->post_connect_accept = post_connect_accept;
274         session->post_connect_terminated = post_connect_terminated;
275         session->post_connect_update = post_connect_update;
276
277         return session;
278 }
279
280 /*
281  * end verse session and free all session data
282  */
283 void end_verse_session(VerseSession *session, char free)
284 {
285         /* send terminate command to verse server */
286         verse_send_connect_terminate(session->address, "blender: bye bye");
287         /* update callbacks */
288         verse_callback_update(1000);
289         /* send destroy session command to verse server */
290         verse_session_destroy(session->vsession);
291         /* set up flag of verse session */
292         session->flag &= ~VERSE_CONNECTED;
293         /* do post connect operations */
294         session->post_connect_terminated(session);
295         /* free session data */
296         free_verse_session_data(session);
297         /* free structure of verse session */
298         if(free) free_verse_session(session);
299 }
300
301 /*
302  * end connection to all verse hosts (servers) ... free all VerseSessions
303  */
304 void end_all_verse_sessions(void)
305 {
306         VerseSession *session;
307
308         session = session_list.first;
309
310         while(session) {
311                 end_verse_session(session, 0);
312                 /* end next session */
313                 session = session->next;
314         }
315
316         BLI_freelistN(&session_list);
317 }
318
319 /*
320  * connect to verse host, set up all callbacks, create session
321  */
322 void b_verse_connect(char *address)
323 {
324         VerseSession *session;
325
326         /* if no session was created before, then set up all callbacks */
327         if((session_list.first==NULL) && (session_list.last==NULL))
328                 set_all_callbacks();
329
330         /* create new session */
331         if(address)
332                 session = create_verse_session("Blender", "pass", address, NULL);
333
334         if(session) {
335                 /* add new session to the list of sessions */
336                 BLI_addtail(&session_list, session);
337
338                 /* add verse handler if this is first session */
339                 if(session_list.first == session_list.last)
340                         add_screenhandler(G.curscreen, SCREEN_HANDLER_VERSE, 1);
341
342         }
343 }
344
345 #endif