parted: Don't return "loop" for non-partitioned devices (RHBZ#634246).
[libguestfs.git] / daemon / sfdisk.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., 675 Mass Ave, Cambridge, MA 02139, 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 <fcntl.h>
26 #include <dirent.h>
27 #include <sys/stat.h>
28
29 #include "daemon.h"
30 #include "actions.h"
31
32 static int
33 sfdisk (const char *device, int n, int cyls, int heads, int sectors,
34         const char *extra_flag,
35         char *const *lines)
36 {
37   FILE *fp;
38   char buf[256];
39   int i;
40
41   strcpy (buf, "sfdisk");
42
43   if (n > 0)
44     sprintf (buf + strlen (buf), " -N %d", n);
45   if (cyls)
46     sprintf (buf + strlen (buf), " -C %d", cyls);
47   if (heads)
48     sprintf (buf + strlen (buf), " -H %d", heads);
49   if (sectors)
50     sprintf (buf + strlen (buf), " -S %d", sectors);
51
52   /* The above are all guaranteed to fit in the fixed-size buffer.
53      However, extra_flag and device have no restrictions,
54      so we must check.  */
55
56   if (extra_flag) {
57     if (strlen (buf) + 1 + strlen (extra_flag) >= sizeof buf) {
58       reply_with_error ("internal buffer overflow: sfdisk extra_flag too long");
59       return -1;
60     }
61     sprintf (buf + strlen (buf), " %s", extra_flag);
62   }
63
64   if (strlen (buf) + 1 + strlen (device) >= sizeof buf) {
65     reply_with_error ("internal buffer overflow: sfdisk device name too long");
66     return -1;
67   }
68   sprintf (buf + strlen (buf), " %s", device);
69
70   if (verbose)
71     printf ("%s\n", buf);
72
73   fp = popen (buf, "w");
74   if (fp == NULL) {
75     reply_with_perror ("failed to open pipe: %s", buf);
76     return -1;
77   }
78
79   for (i = 0; lines[i] != NULL; ++i) {
80     if (fprintf (fp, "%s\n", lines[i]) < 0) {
81       reply_with_perror ("failed to write to pipe: %s", buf);
82       pclose (fp);
83       return -1;
84     }
85   }
86
87   if (pclose (fp) != 0) {
88     reply_with_error ("%s: external command failed", buf);
89     return -1;
90   }
91
92   /* sfdisk sometimes fails on fast machines with:
93    *
94    * Re-reading the partition table ...
95    * BLKRRPART: Device or resource busy
96    * The command to re-read the partition table failed.
97    * Run partprobe(8), kpartx(8) or reboot your system now,
98    * before using mkfs
99    *
100    * Unclear if this is a bug in sfdisk or the kernel or some
101    * other component.  In any case, reread the partition table
102    * unconditionally here.
103    */
104   if (do_blockdev_rereadpt (device) == -1)
105     return -1;
106
107   udev_settle ();
108
109   return 0;
110 }
111
112 int
113 do_sfdisk (const char *device, int cyls, int heads, int sectors,
114            char *const *lines)
115 {
116   return sfdisk (device, 0, cyls, heads, sectors, NULL, lines);
117 }
118
119 int
120 do_sfdisk_N (const char *device, int n, int cyls, int heads, int sectors,
121              const char *line)
122 {
123   char const *const lines[2] = { line, NULL };
124
125   return sfdisk (device, n, cyls, heads, sectors, NULL, (void *) lines);
126 }
127
128 int
129 do_sfdiskM (const char *device, char *const *lines)
130 {
131   return sfdisk (device, 0, 0, 0, 0, "-uM", lines);
132 }
133
134 static char *
135 sfdisk_flag (const char *device, const char *flag)
136 {
137   char *out, *err;
138   int r;
139
140   r = command (&out, &err, "sfdisk", flag, device, NULL);
141   if (r == -1) {
142     reply_with_error ("%s: %s", device, err);
143     free (out);
144     free (err);
145     return NULL;
146   }
147
148   free (err);
149
150   udev_settle ();
151
152   return out;                   /* caller frees */
153 }
154
155 char *
156 do_sfdisk_l (const char *device)
157 {
158   return sfdisk_flag (device, "-l");
159 }
160
161 char *
162 do_sfdisk_kernel_geometry (const char *device)
163 {
164   return sfdisk_flag (device, "-g");
165 }
166
167 char *
168 do_sfdisk_disk_geometry (const char *device)
169 {
170   return sfdisk_flag (device, "-G");
171 }