Rev 4304 | Rev 4422 | Go to most recent revision | Show entire file | Ignore whitespace | Details | Blame | Last modification | View Log | RSS feed
Rev 4304 | Rev 4407 | ||
---|---|---|---|
Line 91... | Line 91... | ||
91 | * Otherwise return 0. |
91 | * Otherwise return 0. |
92 | */ |
92 | */ |
93 | static inline int method_is_forwardable(unative_t method) |
93 | static inline int method_is_forwardable(unative_t method) |
94 | { |
94 | { |
95 | switch (method) { |
95 | switch (method) { |
- | 96 | case IPC_M_CONNECTION_CLONE: |
|
- | 97 | case IPC_M_CONNECT_ME: |
|
96 | case IPC_M_PHONE_HUNGUP: |
98 | case IPC_M_PHONE_HUNGUP: |
97 | /* This message is meant only for the original recipient. */ |
99 | /* This message is meant only for the original recipient. */ |
98 | return 0; |
100 | return 0; |
99 | default: |
101 | default: |
100 | return 1; |
102 | return 1; |
Line 138... | Line 140... | ||
138 | * Return 0 otherwise. |
140 | * Return 0 otherwise. |
139 | */ |
141 | */ |
140 | static inline int answer_need_old(call_t *call) |
142 | static inline int answer_need_old(call_t *call) |
141 | { |
143 | { |
142 | switch (IPC_GET_METHOD(call->data)) { |
144 | switch (IPC_GET_METHOD(call->data)) { |
- | 145 | case IPC_M_CONNECTION_CLONE: |
|
- | 146 | case IPC_M_CONNECT_ME: |
|
143 | case IPC_M_CONNECT_TO_ME: |
147 | case IPC_M_CONNECT_TO_ME: |
144 | case IPC_M_CONNECT_ME_TO: |
148 | case IPC_M_CONNECT_ME_TO: |
145 | case IPC_M_SHARE_OUT: |
149 | case IPC_M_SHARE_OUT: |
146 | case IPC_M_SHARE_IN: |
150 | case IPC_M_SHARE_IN: |
147 | case IPC_M_DATA_WRITE: |
151 | case IPC_M_DATA_WRITE: |
Line 180... | Line 184... | ||
180 | } |
184 | } |
181 | 185 | ||
182 | if (!olddata) |
186 | if (!olddata) |
183 | return 0; |
187 | return 0; |
184 | 188 | ||
- | 189 | if (IPC_GET_METHOD(*olddata) == IPC_M_CONNECTION_CLONE) { |
|
- | 190 | phoneid = IPC_GET_ARG1(*olddata); |
|
- | 191 | phone_t *phone = &TASK->phones[phoneid]; |
|
- | 192 | if (IPC_GET_RETVAL(answer->data) != EOK) { |
|
- | 193 | /* |
|
- | 194 | * The recipient of the cloned phone rejected the offer. |
|
- | 195 | * In this case, the connection was established at the |
|
- | 196 | * request time and therefore we need to slam the phone. |
|
- | 197 | * We don't merely hangup as that would result in |
|
- | 198 | * sending IPC_M_HUNGUP to the third party on the |
|
- | 199 | * other side of the cloned phone. |
|
- | 200 | */ |
|
- | 201 | mutex_lock(&phone->lock); |
|
- | 202 | if (phone->state == IPC_PHONE_CONNECTED) { |
|
- | 203 | spinlock_lock(&phone->callee->lock); |
|
- | 204 | list_remove(&phone->link); |
|
- | 205 | phone->state = IPC_PHONE_SLAMMED; |
|
- | 206 | spinlock_unlock(&phone->callee->lock); |
|
- | 207 | } |
|
- | 208 | mutex_unlock(&phone->lock); |
|
- | 209 | } |
|
- | 210 | } else if (IPC_GET_METHOD(*olddata) == IPC_M_CONNECT_ME) { |
|
- | 211 | phone_t *phone = (phone_t *)IPC_GET_ARG5(*olddata); |
|
- | 212 | if (IPC_GET_RETVAL(answer->data) != EOK) { |
|
- | 213 | /* |
|
- | 214 | * The other party on the cloned phoned rejected our |
|
- | 215 | * request for connection on the protocol level. |
|
- | 216 | * We need to break the connection without sending |
|
- | 217 | * IPC_M_HUNGUP back. |
|
- | 218 | */ |
|
- | 219 | mutex_lock(&phone->lock); |
|
- | 220 | if (phone->state == IPC_PHONE_CONNECTED) { |
|
- | 221 | spinlock_lock(&phone->callee->lock); |
|
- | 222 | list_remove(&phone->link); |
|
- | 223 | phone->state = IPC_PHONE_SLAMMED; |
|
- | 224 | spinlock_unlock(&phone->callee->lock); |
|
- | 225 | } |
|
- | 226 | mutex_unlock(&phone->lock); |
|
- | 227 | } |
|
185 | if (IPC_GET_METHOD(*olddata) == IPC_M_CONNECT_TO_ME) { |
228 | } else if (IPC_GET_METHOD(*olddata) == IPC_M_CONNECT_TO_ME) { |
186 | phoneid = IPC_GET_ARG5(*olddata); |
229 | phoneid = IPC_GET_ARG5(*olddata); |
187 | if (IPC_GET_RETVAL(answer->data)) { |
230 | if (IPC_GET_RETVAL(answer->data) != EOK) { |
188 | /* The connection was not accepted */ |
231 | /* The connection was not accepted */ |
189 | phone_dealloc(phoneid); |
232 | phone_dealloc(phoneid); |
190 | } else { |
233 | } else { |
191 | /* The connection was accepted */ |
234 | /* The connection was accepted */ |
192 | phone_connect(phoneid, &answer->sender->answerbox); |
235 | phone_connect(phoneid, &answer->sender->answerbox); |
Line 194... | Line 237... | ||
194 | IPC_SET_ARG5(answer->data, |
237 | IPC_SET_ARG5(answer->data, |
195 | (unative_t) &TASK->phones[phoneid]); |
238 | (unative_t) &TASK->phones[phoneid]); |
196 | } |
239 | } |
197 | } else if (IPC_GET_METHOD(*olddata) == IPC_M_CONNECT_ME_TO) { |
240 | } else if (IPC_GET_METHOD(*olddata) == IPC_M_CONNECT_ME_TO) { |
198 | /* If the users accepted call, connect */ |
241 | /* If the users accepted call, connect */ |
199 | if (!IPC_GET_RETVAL(answer->data)) { |
242 | if (IPC_GET_RETVAL(answer->data) == EOK) { |
200 | ipc_phone_connect((phone_t *) IPC_GET_ARG5(*olddata), |
243 | ipc_phone_connect((phone_t *) IPC_GET_ARG5(*olddata), |
201 | &TASK->answerbox); |
244 | &TASK->answerbox); |
202 | } |
245 | } |
203 | } else if (IPC_GET_METHOD(*olddata) == IPC_M_SHARE_OUT) { |
246 | } else if (IPC_GET_METHOD(*olddata) == IPC_M_SHARE_OUT) { |
204 | if (!IPC_GET_RETVAL(answer->data)) { |
247 | if (!IPC_GET_RETVAL(answer->data)) { |
Line 306... | Line 349... | ||
306 | size_t size; |
349 | size_t size; |
307 | uintptr_t src; |
350 | uintptr_t src; |
308 | int rc; |
351 | int rc; |
309 | 352 | ||
310 | switch (IPC_GET_METHOD(call->data)) { |
353 | switch (IPC_GET_METHOD(call->data)) { |
- | 354 | case IPC_M_CONNECTION_CLONE: { |
|
- | 355 | phone_t *cloned_phone; |
|
- | 356 | GET_CHECK_PHONE(cloned_phone, IPC_GET_ARG1(call->data), |
|
- | 357 | return ENOENT); |
|
- | 358 | if (cloned_phone < phone) { |
|
- | 359 | mutex_lock(&cloned_phone->lock); |
|
- | 360 | mutex_lock(&phone->lock); |
|
- | 361 | } else { |
|
- | 362 | mutex_lock(&phone->lock); |
|
- | 363 | mutex_lock(&cloned_phone->lock); |
|
- | 364 | } |
|
- | 365 | if ((cloned_phone->state != IPC_PHONE_CONNECTED) || |
|
- | 366 | phone->state != IPC_PHONE_CONNECTED) { |
|
- | 367 | mutex_unlock(&cloned_phone->lock); |
|
- | 368 | mutex_unlock(&phone->lock); |
|
- | 369 | return EINVAL; |
|
- | 370 | } |
|
- | 371 | /* |
|
- | 372 | * We can be pretty sure now that both tasks exist and we are |
|
- | 373 | * connected to them. As we continue to hold the phone locks, |
|
- | 374 | * we are effectively preventing them from finishing their |
|
- | 375 | * potential cleanup. |
|
- | 376 | */ |
|
- | 377 | newphid = phone_alloc(phone->callee->task); |
|
- | 378 | if (newphid < 0) { |
|
- | 379 | mutex_unlock(&cloned_phone->lock); |
|
- | 380 | mutex_unlock(&phone->lock); |
|
- | 381 | return ELIMIT; |
|
- | 382 | } |
|
- | 383 | ipc_phone_connect(&phone->callee->task->phones[newphid], |
|
- | 384 | cloned_phone->callee); |
|
- | 385 | mutex_unlock(&cloned_phone->lock); |
|
- | 386 | mutex_unlock(&phone->lock); |
|
- | 387 | /* Set the new phone for the callee. */ |
|
- | 388 | IPC_SET_ARG1(call->data, newphid); |
|
- | 389 | break; |
|
- | 390 | } |
|
- | 391 | case IPC_M_CONNECT_ME: |
|
- | 392 | IPC_SET_ARG5(call->data, (unative_t) phone); |
|
- | 393 | break; |
|
311 | case IPC_M_CONNECT_ME_TO: |
394 | case IPC_M_CONNECT_ME_TO: |
312 | newphid = phone_alloc(); |
395 | newphid = phone_alloc(TASK); |
313 | if (newphid < 0) |
396 | if (newphid < 0) |
314 | return ELIMIT; |
397 | return ELIMIT; |
315 | /* Set arg5 for server */ |
398 | /* Set arg5 for server */ |
316 | IPC_SET_ARG5(call->data, (unative_t) &TASK->phones[newphid]); |
399 | IPC_SET_ARG5(call->data, (unative_t) &TASK->phones[newphid]); |
317 | call->flags |= IPC_CALL_CONN_ME_TO; |
400 | call->flags |= IPC_CALL_CONN_ME_TO; |
Line 397... | Line 480... | ||
397 | static int process_request(answerbox_t *box, call_t *call) |
480 | static int process_request(answerbox_t *box, call_t *call) |
398 | { |
481 | { |
399 | int phoneid; |
482 | int phoneid; |
400 | 483 | ||
401 | if (IPC_GET_METHOD(call->data) == IPC_M_CONNECT_TO_ME) { |
484 | if (IPC_GET_METHOD(call->data) == IPC_M_CONNECT_TO_ME) { |
402 | phoneid = phone_alloc(); |
485 | phoneid = phone_alloc(TASK); |
403 | if (phoneid < 0) { /* Failed to allocate phone */ |
486 | if (phoneid < 0) { /* Failed to allocate phone */ |
404 | IPC_SET_RETVAL(call->data, ELIMIT); |
487 | IPC_SET_RETVAL(call->data, ELIMIT); |
405 | ipc_answer(box, call); |
488 | ipc_answer(box, call); |
406 | return -1; |
489 | return -1; |
407 | } |
490 | } |