• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /******************************************************************************
2  *
3  *  Copyright (C) 2003-2012 Broadcom Corporation
4  *
5  *  Licensed under the Apache License, Version 2.0 (the "License");
6  *  you may not use this file except in compliance with the License.
7  *  You may obtain a copy of the License at:
8  *
9  *  http://www.apache.org/licenses/LICENSE-2.0
10  *
11  *  Unless required by applicable law or agreed to in writing, software
12  *  distributed under the License is distributed on an "AS IS" BASIS,
13  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  *  See the License for the specific language governing permissions and
15  *  limitations under the License.
16  *
17  ******************************************************************************/
18 
19 /******************************************************************************
20  *
21  *  This module contains API of the audio/video control transport protocol.
22  *
23  ******************************************************************************/
24 
25 #include <string.h>
26 #include "bt_types.h"
27 #include "bt_target.h"
28 #include "bt_utils.h"
29 #include "gki.h"
30 #include "l2c_api.h"
31 #include "l2cdefs.h"
32 #include "btm_api.h"
33 #include "avct_api.h"
34 #include "avct_int.h"
35 
36 /* Control block for AVCT */
37 #if AVCT_DYNAMIC_MEMORY == FALSE
38 tAVCT_CB avct_cb;
39 #endif
40 
41 /*******************************************************************************
42 **
43 ** Function         AVCT_Register
44 **
45 ** Description      This is the system level registration function for the
46 **                  AVCTP protocol.  This function initializes AVCTP and
47 **                  prepares the protocol stack for its use.  This function
48 **                  must be called once by the system or platform using AVCTP
49 **                  before the other functions of the API an be used.
50 **
51 **
52 ** Returns          void
53 **
54 *******************************************************************************/
AVCT_Register(UINT16 mtu,UINT16 mtu_br,UINT8 sec_mask)55 void AVCT_Register(UINT16 mtu, UINT16 mtu_br, UINT8 sec_mask)
56 {
57     UNUSED(mtu_br);
58 
59     AVCT_TRACE_API("AVCT_Register");
60 
61     /* register PSM with L2CAP */
62     L2CA_Register(AVCT_PSM, (tL2CAP_APPL_INFO *) &avct_l2c_appl);
63 
64     /* set security level */
65     BTM_SetSecurityLevel(TRUE, "", BTM_SEC_SERVICE_AVCTP, sec_mask, AVCT_PSM, 0, 0);
66     BTM_SetSecurityLevel(FALSE, "", BTM_SEC_SERVICE_AVCTP, sec_mask, AVCT_PSM, 0, 0);
67 
68     /* initialize AVCTP data structures */
69     memset(&avct_cb, 0, sizeof(tAVCT_CB));
70 
71 #if (AVCT_BROWSE_INCLUDED == TRUE)
72     /* Include the browsing channel which uses eFCR */
73     L2CA_Register(AVCT_BR_PSM, (tL2CAP_APPL_INFO *) &avct_l2c_br_appl);
74 
75     BTM_SetSecurityLevel(TRUE, "", BTM_SEC_SERVICE_AVCTP_BROWSE, sec_mask, AVCT_BR_PSM, 0, 0);
76     BTM_SetSecurityLevel(FALSE, "", BTM_SEC_SERVICE_AVCTP_BROWSE, sec_mask, AVCT_BR_PSM, 0, 0);
77 
78     if (mtu_br < AVCT_MIN_BROWSE_MTU)
79         mtu_br = AVCT_MIN_BROWSE_MTU;
80     avct_cb.mtu_br = mtu_br;
81 #endif
82 
83 #if defined(AVCT_INITIAL_TRACE_LEVEL)
84     avct_cb.trace_level = AVCT_INITIAL_TRACE_LEVEL;
85 #else
86     avct_cb.trace_level = BT_TRACE_LEVEL_NONE;
87 #endif
88 
89     if (mtu < AVCT_MIN_CONTROL_MTU)
90         mtu = AVCT_MIN_CONTROL_MTU;
91     /* store mtu */
92     avct_cb.mtu = mtu;
93 }
94 
95 /*******************************************************************************
96 **
97 ** Function         AVCT_Deregister
98 **
99 ** Description      This function is called to deregister use AVCTP protocol.
100 **                  It is called when AVCTP is no longer being used by any
101 **                  application in the system.  Before this function can be
102 **                  called, all connections must be removed with
103 **                  AVCT_RemoveConn().
104 **
105 **
106 ** Returns          void
107 **
108 *******************************************************************************/
AVCT_Deregister(void)109 void AVCT_Deregister(void)
110 {
111     AVCT_TRACE_API("AVCT_Deregister");
112 
113     /* deregister PSM with L2CAP */
114     L2CA_Deregister(AVCT_PSM);
115 }
116 
117 /*******************************************************************************
118 **
119 ** Function         AVCT_CreateConn
120 **
121 ** Description      Create an AVCTP connection.  There are two types of
122 **                  connections, initiator and acceptor, as determined by
123 **                  the p_cc->role parameter.  When this function is called to
124 **                  create an initiator connection, an AVCTP connection to
125 **                  the peer device is initiated if one does not already exist.
126 **                  If an acceptor connection is created, the connection waits
127 **                  passively for an incoming AVCTP connection from a peer device.
128 **
129 **
130 ** Returns          AVCT_SUCCESS if successful, otherwise error.
131 **
132 *******************************************************************************/
AVCT_CreateConn(UINT8 * p_handle,tAVCT_CC * p_cc,BD_ADDR peer_addr)133 UINT16 AVCT_CreateConn(UINT8 *p_handle, tAVCT_CC *p_cc, BD_ADDR peer_addr)
134 {
135     UINT16      result = AVCT_SUCCESS;
136     tAVCT_CCB   *p_ccb;
137     tAVCT_LCB   *p_lcb;
138 
139     AVCT_TRACE_API("AVCT_CreateConn: %d, control:%d", p_cc->role, p_cc->control);
140 
141     /* Allocate ccb; if no ccbs, return failure */
142     if ((p_ccb = avct_ccb_alloc(p_cc)) == NULL)
143     {
144         result = AVCT_NO_RESOURCES;
145     }
146     else
147     {
148         /* get handle */
149         *p_handle = avct_ccb_to_idx(p_ccb);
150 
151         /* if initiator connection */
152         if (p_cc->role == AVCT_INT)
153         {
154             /* find link; if none allocate a new one */
155             if ((p_lcb = avct_lcb_by_bd(peer_addr)) == NULL)
156             {
157                 if ((p_lcb = avct_lcb_alloc(peer_addr)) == NULL)
158                 {
159                     /* no link resources; free ccb as well */
160                     avct_ccb_dealloc(p_ccb, AVCT_NO_EVT, 0, NULL);
161                     result = AVCT_NO_RESOURCES;
162                 }
163             }
164             /* check if PID already in use */
165             else if (avct_lcb_has_pid(p_lcb, p_cc->pid))
166             {
167                 avct_ccb_dealloc(p_ccb, AVCT_NO_EVT, 0, NULL);
168                 result = AVCT_PID_IN_USE;
169             }
170 
171             if (result == AVCT_SUCCESS)
172             {
173                 /* bind lcb to ccb */
174                 p_ccb->p_lcb = p_lcb;
175                 AVCT_TRACE_DEBUG("ch_state: %d", p_lcb->ch_state);
176                 avct_lcb_event(p_lcb, AVCT_LCB_UL_BIND_EVT, (tAVCT_LCB_EVT *) &p_ccb);
177             }
178         }
179     }
180     return result;
181 }
182 
183 /*******************************************************************************
184 **
185 ** Function         AVCT_RemoveConn
186 **
187 ** Description      Remove an AVCTP connection.  This function is called when
188 **                  the application is no longer using a connection.  If this
189 **                  is the last connection to a peer the L2CAP channel for AVCTP
190 **                  will be closed.
191 **
192 **
193 ** Returns          AVCT_SUCCESS if successful, otherwise error.
194 **
195 *******************************************************************************/
AVCT_RemoveConn(UINT8 handle)196 UINT16 AVCT_RemoveConn(UINT8 handle)
197 {
198     UINT16              result = AVCT_SUCCESS;
199     tAVCT_CCB           *p_ccb;
200 
201     AVCT_TRACE_API("AVCT_RemoveConn");
202 
203     /* map handle to ccb */
204     if ((p_ccb = avct_ccb_by_idx(handle)) == NULL)
205     {
206         result = AVCT_BAD_HANDLE;
207     }
208     /* if connection not bound to lcb, dealloc */
209     else if (p_ccb->p_lcb == NULL)
210     {
211         avct_ccb_dealloc(p_ccb, AVCT_NO_EVT, 0, NULL);
212     }
213     /* send unbind event to lcb */
214     else
215     {
216         avct_lcb_event(p_ccb->p_lcb, AVCT_LCB_UL_UNBIND_EVT, (tAVCT_LCB_EVT *) &p_ccb);
217     }
218     return result;
219 }
220 
221 /*******************************************************************************
222 **
223 ** Function         AVCT_CreateBrowse
224 **
225 ** Description      Create an AVCTP Browse channel.  There are two types of
226 **                  connections, initiator and acceptor, as determined by
227 **                  the role parameter.  When this function is called to
228 **                  create an initiator connection, the Browse channel to
229 **                  the peer device is initiated if one does not already exist.
230 **                  If an acceptor connection is created, the connection waits
231 **                  passively for an incoming AVCTP connection from a peer device.
232 **
233 **
234 ** Returns          AVCT_SUCCESS if successful, otherwise error.
235 **
236 *******************************************************************************/
AVCT_CreateBrowse(UINT8 handle,UINT8 role)237 UINT16 AVCT_CreateBrowse (UINT8 handle, UINT8 role)
238 {
239 #if (AVCT_BROWSE_INCLUDED == TRUE)
240     UINT16      result = AVCT_SUCCESS;
241     tAVCT_CCB   *p_ccb;
242     tAVCT_BCB   *p_bcb;
243     int         index;
244 
245     AVCT_TRACE_API("AVCT_CreateBrowse: %d", role);
246 
247     /* map handle to ccb */
248     if ((p_ccb = avct_ccb_by_idx(handle)) == NULL)
249     {
250         return AVCT_BAD_HANDLE;
251     }
252     else
253     {
254         /* mark this CCB as supporting browsing channel */
255         if ((p_ccb->allocated & AVCT_ALOC_BCB) == 0)
256         {
257             p_ccb->allocated |= AVCT_ALOC_BCB;
258         }
259     }
260 
261     /* if initiator connection */
262     if (role == AVCT_INT)
263     {
264         /* the link control block must exist before this function is called as INT. */
265         if ((p_ccb->p_lcb == NULL) || (p_ccb->p_lcb->allocated == 0))
266         {
267             result = AVCT_NOT_OPEN;
268         }
269         else
270         {
271             /* find link; if none allocate a new one */
272             index = p_ccb->p_lcb->allocated;
273             if (index > AVCT_NUM_LINKS)
274             {
275                 result = AVCT_BAD_HANDLE;
276             }
277             else
278             {
279                 p_bcb = &avct_cb.bcb[index - 1];
280                 p_bcb->allocated = index;
281             }
282         }
283 
284         if (result == AVCT_SUCCESS)
285         {
286             /* bind bcb to ccb */
287             p_ccb->p_bcb = p_bcb;
288             AVCT_TRACE_DEBUG("ch_state: %d", p_bcb->ch_state);
289             avct_bcb_event(p_bcb, AVCT_LCB_UL_BIND_EVT, (tAVCT_LCB_EVT *) &p_ccb);
290         }
291     }
292 
293     return result;
294 #else
295     UNUSED(handle);
296     UNUSED(role);
297     return AVCT_NO_RESOURCES;
298 #endif
299 }
300 
301 /*******************************************************************************
302 **
303 ** Function         AVCT_RemoveBrowse
304 **
305 ** Description      Remove an AVCTP Browse channel.  This function is called when
306 **                  the application is no longer using a connection.  If this
307 **                  is the last connection to a peer the L2CAP channel for AVCTP
308 **                  will be closed.
309 **
310 **
311 ** Returns          AVCT_SUCCESS if successful, otherwise error.
312 **
313 *******************************************************************************/
AVCT_RemoveBrowse(UINT8 handle)314 UINT16 AVCT_RemoveBrowse (UINT8 handle)
315 {
316 #if (AVCT_BROWSE_INCLUDED == TRUE)
317     UINT16              result = AVCT_SUCCESS;
318     tAVCT_CCB           *p_ccb;
319 
320     AVCT_TRACE_API("AVCT_RemoveBrowse");
321 
322     /* map handle to ccb */
323     if ((p_ccb = avct_ccb_by_idx(handle)) == NULL)
324     {
325         result = AVCT_BAD_HANDLE;
326     }
327     else if (p_ccb->p_bcb != NULL)
328     /* send unbind event to bcb */
329     {
330         avct_bcb_event(p_ccb->p_bcb, AVCT_LCB_UL_UNBIND_EVT, (tAVCT_LCB_EVT *) &p_ccb);
331     }
332     return result;
333 #else
334     UNUSED(handle);
335     return AVCT_NO_RESOURCES;
336 #endif
337 }
338 
339 /*******************************************************************************
340 **
341 ** Function         AVCT_GetBrowseMtu
342 **
343 ** Description      Get the peer_mtu for the AVCTP Browse channel of the given
344 **                  connection.
345 **
346 ** Returns          the peer browsing channel MTU.
347 **
348 *******************************************************************************/
AVCT_GetBrowseMtu(UINT8 handle)349 UINT16 AVCT_GetBrowseMtu (UINT8 handle)
350 {
351     UINT16  peer_mtu = AVCT_MIN_BROWSE_MTU;
352 #if (AVCT_BROWSE_INCLUDED == TRUE)
353     tAVCT_CCB           *p_ccb;
354 
355     if ((p_ccb = avct_ccb_by_idx(handle)) != NULL && p_ccb->p_bcb != NULL)
356     {
357         peer_mtu = p_ccb->p_bcb->peer_mtu;
358     }
359 #else
360     UNUSED(handle);
361 #endif
362     return peer_mtu;
363 }
364 
365 /*******************************************************************************
366 **
367 ** Function         AVCT_GetPeerMtu
368 **
369 ** Description      Get the peer_mtu for the AVCTP channel of the given
370 **                  connection.
371 **
372 ** Returns          the peer MTU size.
373 **
374 *******************************************************************************/
AVCT_GetPeerMtu(UINT8 handle)375 UINT16 AVCT_GetPeerMtu (UINT8 handle)
376 {
377     UINT16      peer_mtu = L2CAP_DEFAULT_MTU;
378     tAVCT_CCB   *p_ccb;
379 
380     /* map handle to ccb */
381     if ((p_ccb = avct_ccb_by_idx(handle)) != NULL)
382     {
383         if (p_ccb->p_lcb)
384         {
385             peer_mtu = p_ccb->p_lcb->peer_mtu;
386         }
387     }
388 
389     return peer_mtu;
390 }
391 
392 /*******************************************************************************
393 **
394 ** Function         AVCT_MsgReq
395 **
396 ** Description      Send an AVCTP message to a peer device.  In calling
397 **                  AVCT_MsgReq(), the application should keep track of the
398 **                  congestion state of AVCTP as communicated with events
399 **                  AVCT_CONG_IND_EVT and AVCT_UNCONG_IND_EVT.   If the
400 **                  application calls AVCT_MsgReq() when AVCTP is congested
401 **                  the message may be discarded.  The application may make its
402 **                  first call to AVCT_MsgReq() after it receives an
403 **                  AVCT_CONNECT_CFM_EVT or AVCT_CONNECT_IND_EVT on control channel or
404 **                  AVCT_BROWSE_CONN_CFM_EVT or AVCT_BROWSE_CONN_IND_EVT on browsing channel.
405 **
406 **                  p_msg->layer_specific must be set to
407 **                  AVCT_DATA_CTRL for control channel traffic;
408 **                  AVCT_DATA_BROWSE for for browse channel traffic.
409 **
410 ** Returns          AVCT_SUCCESS if successful, otherwise error.
411 **
412 *******************************************************************************/
AVCT_MsgReq(UINT8 handle,UINT8 label,UINT8 cr,BT_HDR * p_msg)413 UINT16 AVCT_MsgReq(UINT8 handle, UINT8 label, UINT8 cr, BT_HDR *p_msg)
414 {
415     UINT16          result = AVCT_SUCCESS;
416     tAVCT_CCB       *p_ccb;
417     tAVCT_UL_MSG    ul_msg;
418 
419     AVCT_TRACE_API("AVCT_MsgReq");
420 
421     /* verify p_msg parameter */
422     if (p_msg == NULL)
423     {
424         return AVCT_NO_RESOURCES;
425     }
426     AVCT_TRACE_API("len: %d", p_msg->len);
427 
428     /* map handle to ccb */
429     if ((p_ccb = avct_ccb_by_idx(handle)) == NULL)
430     {
431         result = AVCT_BAD_HANDLE;
432         GKI_freebuf(p_msg);
433     }
434     /* verify channel is bound to link */
435     else if (p_ccb->p_lcb == NULL)
436     {
437         result = AVCT_NOT_OPEN;
438         GKI_freebuf(p_msg);
439     }
440 
441     if (result == AVCT_SUCCESS)
442     {
443         ul_msg.p_buf = p_msg;
444         ul_msg.p_ccb = p_ccb;
445         ul_msg.label = label;
446         ul_msg.cr = cr;
447 
448 #if (AVCT_BROWSE_INCLUDED == TRUE)
449         /* send msg event to bcb */
450         if (p_msg->layer_specific == AVCT_DATA_BROWSE)
451         {
452             if (p_ccb->p_bcb == NULL && (p_ccb->allocated & AVCT_ALOC_BCB) == 0)
453             {
454                 /* BCB channel is not open and not allocated */
455                 result = AVCT_BAD_HANDLE;
456                 GKI_freebuf(p_msg);
457             }
458             else
459             {
460                 p_ccb->p_bcb = avct_bcb_by_lcb(p_ccb->p_lcb);
461                 avct_bcb_event(p_ccb->p_bcb, AVCT_LCB_UL_MSG_EVT, (tAVCT_LCB_EVT *) &ul_msg);
462             }
463         }
464         /* send msg event to lcb */
465         else
466 #endif
467         {
468             avct_lcb_event(p_ccb->p_lcb, AVCT_LCB_UL_MSG_EVT, (tAVCT_LCB_EVT *) &ul_msg);
469         }
470     }
471     return result;
472 }
473 
474