]> sigrok.org Git - libserialport.git/blob - macosx.c
Build: Include config.h first in all source files
[libserialport.git] / macosx.c
1 /*
2  * This file is part of the libserialport project.
3  *
4  * Copyright (C) 2013-2014 Martin Ling <martin-libserialport@earth.li>
5  * Copyright (C) 2014 Aurelien Jacobs <aurel@gnuage.org>
6  *
7  * This program is free software: you can redistribute it and/or modify
8  * it under the terms of the GNU Lesser General Public License as
9  * published by the Free Software Foundation, either version 3 of the
10  * License, or (at your option) any later version.
11  *
12  * This program 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
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public License
18  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
19  */
20
21 #include <config.h>
22 #include "libserialport.h"
23 #include "libserialport_internal.h"
24
25 SP_PRIV enum sp_return get_port_details(struct sp_port *port)
26 {
27         /*
28          * Description limited to 127 char, anything longer
29          * would not be user friendly anyway.
30          */
31         char description[128];
32         int bus, address, vid, pid = -1;
33         char manufacturer[128], product[128], serial[128];
34         CFMutableDictionaryRef classes;
35         io_iterator_t iter;
36         io_object_t ioport, ioparent;
37         CFTypeRef cf_property, cf_bus, cf_address, cf_vendor, cf_product;
38         Boolean result;
39         char path[PATH_MAX], class[16];
40
41         DEBUG("Getting serial port list");
42         if (!(classes = IOServiceMatching(kIOSerialBSDServiceValue)))
43                 RETURN_FAIL("IOServiceMatching() failed");
44
45         if (IOServiceGetMatchingServices(kIOMasterPortDefault, classes,
46                                          &iter) != KERN_SUCCESS)
47                 RETURN_FAIL("IOServiceGetMatchingServices() failed");
48
49         DEBUG("Iterating over results");
50         while ((ioport = IOIteratorNext(iter))) {
51                 if (!(cf_property = IORegistryEntryCreateCFProperty(ioport,
52                             CFSTR(kIOCalloutDeviceKey), kCFAllocatorDefault, 0))) {
53                         IOObjectRelease(ioport);
54                         continue;
55                 }
56                 result = CFStringGetCString(cf_property, path, sizeof(path),
57                                             kCFStringEncodingASCII);
58                 CFRelease(cf_property);
59                 if (!result || strcmp(path, port->name)) {
60                         IOObjectRelease(ioport);
61                         continue;
62                 }
63                 DEBUG_FMT("Found port %s", path);
64
65                 IORegistryEntryGetParentEntry(ioport, kIOServicePlane, &ioparent);
66                 if ((cf_property=IORegistryEntrySearchCFProperty(ioparent,kIOServicePlane,
67                            CFSTR("IOProviderClass"), kCFAllocatorDefault,
68                            kIORegistryIterateRecursively | kIORegistryIterateParents))) {
69                         if (CFStringGetCString(cf_property, class, sizeof(class),
70                                                kCFStringEncodingASCII) &&
71                             strstr(class, "USB")) {
72                                 DEBUG("Found USB class device");
73                                 port->transport = SP_TRANSPORT_USB;
74                         }
75                         CFRelease(cf_property);
76                 }
77                 IOObjectRelease(ioparent);
78
79                 if ((cf_property = IORegistryEntrySearchCFProperty(ioport,kIOServicePlane,
80                          CFSTR("USB Interface Name"), kCFAllocatorDefault,
81                          kIORegistryIterateRecursively | kIORegistryIterateParents)) ||
82                     (cf_property = IORegistryEntrySearchCFProperty(ioport,kIOServicePlane,
83                          CFSTR("USB Product Name"), kCFAllocatorDefault,
84                          kIORegistryIterateRecursively | kIORegistryIterateParents)) ||
85                     (cf_property = IORegistryEntrySearchCFProperty(ioport,kIOServicePlane,
86                          CFSTR("Product Name"), kCFAllocatorDefault,
87                          kIORegistryIterateRecursively | kIORegistryIterateParents)) ||
88                     (cf_property = IORegistryEntryCreateCFProperty(ioport,
89                          CFSTR(kIOTTYDeviceKey), kCFAllocatorDefault, 0))) {
90                         if (CFStringGetCString(cf_property, description, sizeof(description),
91                                                kCFStringEncodingASCII)) {
92                                 DEBUG_FMT("Found description %s", description);
93                                 port->description = strdup(description);
94                         }
95                         CFRelease(cf_property);
96                 } else {
97                         DEBUG("No description for this device");
98                 }
99
100                 cf_bus = IORegistryEntrySearchCFProperty(ioport, kIOServicePlane,
101                                                          CFSTR("USBBusNumber"),
102                                                          kCFAllocatorDefault,
103                                                          kIORegistryIterateRecursively
104                                                          | kIORegistryIterateParents);
105                 cf_address = IORegistryEntrySearchCFProperty(ioport, kIOServicePlane,
106                                                          CFSTR("USB Address"),
107                                                          kCFAllocatorDefault,
108                                                          kIORegistryIterateRecursively
109                                                          | kIORegistryIterateParents);
110                 if (cf_bus && cf_address &&
111                     CFNumberGetValue(cf_bus    , kCFNumberIntType, &bus) &&
112                     CFNumberGetValue(cf_address, kCFNumberIntType, &address)) {
113                         DEBUG_FMT("Found matching USB bus:address %03d:%03d", bus, address);
114                         port->usb_bus = bus;
115                         port->usb_address = address;
116                 }
117                 if (cf_bus)
118                         CFRelease(cf_bus);
119                 if (cf_address)
120                         CFRelease(cf_address);
121
122                 cf_vendor = IORegistryEntrySearchCFProperty(ioport, kIOServicePlane,
123                                                          CFSTR("idVendor"),
124                                                          kCFAllocatorDefault,
125                                                          kIORegistryIterateRecursively
126                                                          | kIORegistryIterateParents);
127                 cf_product = IORegistryEntrySearchCFProperty(ioport, kIOServicePlane,
128                                                          CFSTR("idProduct"),
129                                                          kCFAllocatorDefault,
130                                                          kIORegistryIterateRecursively
131                                                          | kIORegistryIterateParents);
132                 if (cf_vendor && cf_product &&
133                     CFNumberGetValue(cf_vendor , kCFNumberIntType, &vid) &&
134                     CFNumberGetValue(cf_product, kCFNumberIntType, &pid)) {
135                         DEBUG_FMT("Found matching USB VID:PID %04X:%04X", vid, pid);
136                         port->usb_vid = vid;
137                         port->usb_pid = pid;
138                 }
139                 if (cf_vendor)
140                         CFRelease(cf_vendor);
141                 if (cf_product)
142                         CFRelease(cf_product);
143
144                 if ((cf_property = IORegistryEntrySearchCFProperty(ioport,kIOServicePlane,
145                          CFSTR("USB Vendor Name"), kCFAllocatorDefault,
146                          kIORegistryIterateRecursively | kIORegistryIterateParents))) {
147                         if (CFStringGetCString(cf_property, manufacturer, sizeof(manufacturer),
148                                                kCFStringEncodingASCII)) {
149                                 DEBUG_FMT("Found manufacturer %s", manufacturer);
150                                 port->usb_manufacturer = strdup(manufacturer);
151                         }
152                         CFRelease(cf_property);
153                 }
154
155                 if ((cf_property = IORegistryEntrySearchCFProperty(ioport,kIOServicePlane,
156                          CFSTR("USB Product Name"), kCFAllocatorDefault,
157                          kIORegistryIterateRecursively | kIORegistryIterateParents))) {
158                         if (CFStringGetCString(cf_property, product, sizeof(product),
159                                                kCFStringEncodingASCII)) {
160                                 DEBUG_FMT("Found product name %s", product);
161                                 port->usb_product = strdup(product);
162                         }
163                         CFRelease(cf_property);
164                 }
165
166                 if ((cf_property = IORegistryEntrySearchCFProperty(ioport,kIOServicePlane,
167                          CFSTR("USB Serial Number"), kCFAllocatorDefault,
168                          kIORegistryIterateRecursively | kIORegistryIterateParents))) {
169                         if (CFStringGetCString(cf_property, serial, sizeof(serial),
170                                                kCFStringEncodingASCII)) {
171                                 DEBUG_FMT("Found serial number %s", serial);
172                                 port->usb_serial = strdup(serial);
173                         }
174                         CFRelease(cf_property);
175                 }
176
177                 IOObjectRelease(ioport);
178                 break;
179         }
180         IOObjectRelease(iter);
181
182         RETURN_OK();
183 }
184
185 SP_PRIV enum sp_return list_ports(struct sp_port ***list)
186 {
187         CFMutableDictionaryRef classes;
188         io_iterator_t iter;
189         char path[PATH_MAX];
190         io_object_t port;
191         CFTypeRef cf_path;
192         Boolean result;
193         int ret = SP_OK;
194
195         DEBUG("Creating matching dictionary");
196         if (!(classes = IOServiceMatching(kIOSerialBSDServiceValue))) {
197                 SET_FAIL(ret, "IOServiceMatching() failed");
198                 goto out_done;
199         }
200
201         DEBUG("Getting matching services");
202         if (IOServiceGetMatchingServices(kIOMasterPortDefault, classes,
203                                          &iter) != KERN_SUCCESS) {
204                 SET_FAIL(ret, "IOServiceGetMatchingServices() failed");
205                 goto out_done;
206         }
207
208         DEBUG("Iterating over results");
209         while ((port = IOIteratorNext(iter))) {
210                 cf_path = IORegistryEntryCreateCFProperty(port,
211                                 CFSTR(kIOCalloutDeviceKey), kCFAllocatorDefault, 0);
212                 if (cf_path) {
213                         result = CFStringGetCString(cf_path, path, sizeof(path),
214                                                     kCFStringEncodingASCII);
215                         CFRelease(cf_path);
216                         if (result) {
217                                 DEBUG_FMT("Found port %s", path);
218                                 if (!(*list = list_append(*list, path))) {
219                                         SET_ERROR(ret, SP_ERR_MEM, "List append failed");
220                                         IOObjectRelease(port);
221                                         goto out;
222                                 }
223                         }
224                 }
225                 IOObjectRelease(port);
226         }
227 out:
228         IOObjectRelease(iter);
229 out_done:
230
231         return ret;
232 }