c610a41822fb93f27a137402fd552ae32f99ff66
[libguestfs.git] / daemon / blkid.c
1 /* libguestfs - the guestfsd daemon
2  * Copyright (C) 2009 Red Hat Inc.
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
17  */
18
19 #include <config.h>
20
21 #include <stdio.h>
22 #include <stdlib.h>
23 #include <string.h>
24 #include <unistd.h>
25 #include <limits.h>
26
27 #include "daemon.h"
28 #include "actions.h"
29
30 static char *
31 get_blkid_tag (const char *device, const char *tag)
32 {
33   char *out, *err;
34   int r;
35
36   r = commandr (&out, &err,
37                 "blkid",
38                 /* Adding -c option kills all caching, even on RHEL 5. */
39                 "-c", "/dev/null",
40                 "-o", "value", "-s", tag, device, NULL);
41   if (r != 0 && r != 2) {
42     if (r >= 0)
43       reply_with_error ("%s: %s (blkid returned %d)", device, err, r);
44     else
45       reply_with_error ("%s: %s", device, err);
46     free (out);
47     free (err);
48     return NULL;
49   }
50
51   free (err);
52
53   if (r == 2) {                 /* means UUID etc not found */
54     free (out);
55     out = strdup ("");
56     if (out == NULL)
57       reply_with_perror ("strdup");
58     return out;
59   }
60
61   /* Trim trailing \n if present. */
62   size_t len = strlen (out);
63   if (len > 0 && out[len-1] == '\n')
64     out[len-1] = '\0';
65
66   return out;                   /* caller frees */
67 }
68
69 char *
70 do_vfs_type (const char *device)
71 {
72   return get_blkid_tag (device, "TYPE");
73 }
74
75 char *
76 do_vfs_label (const char *device)
77 {
78   return get_blkid_tag (device, "LABEL");
79 }
80
81 char *
82 do_vfs_uuid (const char *device)
83 {
84   return get_blkid_tag (device, "UUID");
85 }
86
87 char **
88 do_blkid(const char *device)
89 {
90   int r;
91   char *out = NULL, *err = NULL;
92   char **lines = NULL;
93
94   char **ret = NULL;
95   int size = 0, alloc = 0;
96
97   r = command (&out, &err,
98                "blkid",
99                /* Adding -c option kills all caching, even on RHEL 5. */
100                "-c", "/dev/null",
101                "-p", "-i", "-o", "export", device, NULL);
102   if (r == -1) {
103     reply_with_error("%s", err);
104     goto error;
105   }
106
107   /* Split the command output into lines */
108   lines = split_lines(out);
109   if (lines == NULL) {
110     reply_with_perror("malloc");
111     goto error;
112   }
113
114   /* Parse the output of blkid -p -i -o export:
115    * UUID=b6d83437-c6b4-4bf0-8381-ef3fc3578590
116    * VERSION=1.0
117    * TYPE=ext2
118    * USAGE=filesystem
119    * MINIMUM_IO_SIZE=512
120    * PHYSICAL_SECTOR_SIZE=512
121    * LOGICAL_SECTOR_SIZE=512
122    * PART_ENTRY_SCHEME=dos
123    * PART_ENTRY_TYPE=0x83
124    * PART_ENTRY_NUMBER=6
125    * PART_ENTRY_OFFSET=642875153
126    * PART_ENTRY_SIZE=104857600
127    * PART_ENTRY_DISK=8:0
128    */
129   for (char **i = lines; *i != NULL; i++) {
130     char *line = *i;
131
132     /* Skip blank lines (shouldn't happen) */
133     if (line[0] == '\0') continue;
134
135     /* Split the line in 2 at the equals sign */
136     char *eq = strchr(line, '=');
137     if (eq) {
138       *eq = '\0'; eq++;
139
140       /* Add the key/value pair to the output */
141       if (add_string(&ret, &size, &alloc, line) == -1 ||
142           add_string(&ret, &size, &alloc, eq) == -1) goto error;
143     } else {
144       fprintf(stderr, "blkid: unexpected blkid output ignored: %s", line);
145     }
146   }
147
148   free(out);
149   free(err);
150   free(lines);
151
152   if (add_string(&ret, &size, &alloc, NULL) == -1) return NULL;
153
154   return ret;
155
156 error:
157   free(out);
158   free(err);
159   if (lines) free(lines);
160   if (ret) free_strings(ret);
161
162   return NULL;
163 }