Update module-bluetooth-device to the new ipc.
[profile/ivi/pulseaudio-panda.git] / src / modules / bluetooth / ipc.c
1 /*
2  *
3  *  BlueZ - Bluetooth protocol stack for Linux
4  *
5  *  Copyright (C) 2004-2008  Marcel Holtmann <marcel@holtmann.org>
6  *
7  *  This library is free software; you can redistribute it and/or
8  *  modify it under the terms of the GNU Lesser General Public
9  *  License as published by the Free Software Foundation; either
10  *  version 2.1 of the License, or (at your option) any later version.
11  *
12  *  This library is distributed in the hope that it will be useful,
13  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
14  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  *  Lesser General Public License for more details.
16  *
17  *  You should have received a copy of the GNU Lesser General Public
18  *  License along with this library; if not, write to the Free Software
19  *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
20  *
21  */
22
23 #include "ipc.h"
24
25 #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
26
27 /* This table contains the string representation for messages types */
28 static const char *strtypes[] = {
29         "BT_REQUEST",
30         "BT_RESPONSE",
31         "BT_INDICATION",
32         "BT_ERROR",
33 };
34
35 /* This table contains the string representation for messages names */
36 static const char *strnames[] = {
37         "BT_GET_CAPABILITIES",
38         "BT_SET_CONFIGURATION",
39         "BT_NEW_STREAM",
40         "BT_START_STREAM",
41         "BT_STOP_STREAM",
42         "BT_SUSPEND_STREAM",
43         "BT_RESUME_STREAM",
44         "BT_CONTROL",
45 };
46
47 int bt_audio_service_open(void)
48 {
49         int sk;
50         int err;
51         struct sockaddr_un addr = {
52                 AF_UNIX, BT_IPC_SOCKET_NAME
53         };
54
55         sk = socket(PF_LOCAL, SOCK_STREAM, 0);
56         if (sk < 0) {
57                 err = errno;
58                 fprintf(stderr, "%s: Cannot open socket: %s (%d)\n",
59                         __FUNCTION__, strerror(err), err);
60                 errno = err;
61                 return -1;
62         }
63
64         if (connect(sk, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
65                 err = errno;
66                 fprintf(stderr, "%s: connect() failed: %s (%d)\n",
67                         __FUNCTION__, strerror(err), err);
68                 close(sk);
69                 errno = err;
70                 return -1;
71         }
72
73         return sk;
74 }
75
76 int bt_audio_service_close(int sk)
77 {
78         return close(sk);
79 }
80
81 int bt_audio_service_get_data_fd(int sk)
82 {
83         char cmsg_b[CMSG_SPACE(sizeof(int))], m;
84         int err, ret;
85         struct iovec iov = { &m, sizeof(m) };
86         struct msghdr msgh;
87         struct cmsghdr *cmsg;
88
89         memset(&msgh, 0, sizeof(msgh));
90         msgh.msg_iov = &iov;
91         msgh.msg_iovlen = 1;
92         msgh.msg_control = &cmsg_b;
93         msgh.msg_controllen = CMSG_LEN(sizeof(int));
94
95         ret = recvmsg(sk, &msgh, 0);
96         if (ret < 0) {
97                 err = errno;
98                 fprintf(stderr, "%s: Unable to receive fd: %s (%d)\n",
99                         __FUNCTION__, strerror(err), err);
100                 errno = err;
101                 return -1;
102         }
103
104         /* Receive auxiliary data in msgh */
105         for (cmsg = CMSG_FIRSTHDR(&msgh); cmsg != NULL;
106                         cmsg = CMSG_NXTHDR(&msgh, cmsg)) {
107                 if (cmsg->cmsg_level == SOL_SOCKET
108                                 && cmsg->cmsg_type == SCM_RIGHTS)
109                         return (*(int *) CMSG_DATA(cmsg));
110         }
111
112         errno = EINVAL;
113         return -1;
114 }
115
116 const char *bt_audio_strtype(uint8_t type)
117 {
118         if (type >= ARRAY_SIZE(strtypes))
119                 return NULL;
120
121         return strtypes[type];
122 }
123
124 const char *bt_audio_strname(uint8_t name)
125 {
126         if (name >= ARRAY_SIZE(strnames))
127                 return NULL;
128
129         return strnames[name];
130 }