blob: e1953a9aa1803c32708a905e42b2e8fce153d897 [file] [log] [blame]
Venkateswararao Jujjuri (JV)86e42d72011-08-08 23:33:48 +05301
2/*
3 * Virtio 9p backend
4 *
5 * Copyright IBM, Corp. 2011
6 *
7 * Authors:
8 * Aneesh Kumar K.V <aneesh.kumar@linux.vnet.ibm.com>
9 *
10 * This work is licensed under the terms of the GNU GPL, version 2. See
11 * the COPYING file in the top-level directory.
12 *
13 */
14
15#include "fsdev/qemu-fsdev.h"
Paolo Bonzini1de7afc2012-12-17 18:20:00 +010016#include "qemu/thread.h"
Daniel P. Berrange10817bf2015-09-01 14:48:02 +010017#include "qemu/coroutine.h"
Venkateswararao Jujjuri (JV)86e42d72011-08-08 23:33:48 +053018#include "virtio-9p-coth.h"
19
Chen Gang4fa4ce72014-03-02 01:36:19 +080020static ssize_t __readlink(V9fsState *s, V9fsPath *path, V9fsString *buf)
21{
22 ssize_t len, maxlen = PATH_MAX;
23
24 buf->data = g_malloc(PATH_MAX);
25 for(;;) {
26 len = s->ops->readlink(&s->ctx, path, buf->data, maxlen);
27 if (len < 0) {
28 g_free(buf->data);
29 buf->data = NULL;
30 buf->size = 0;
31 break;
32 } else if (len == maxlen) {
33 /*
34 * We dodn't have space to put the NULL or we have more
35 * to read. Increase the size and try again
36 */
37 maxlen *= 2;
38 g_free(buf->data);
39 buf->data = g_malloc(maxlen);
40 continue;
41 }
42 /*
43 * Null terminate the readlink output
44 */
45 buf->data[len] = '\0';
46 buf->size = len;
47 break;
48 }
49 return len;
50}
51
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053052int v9fs_co_readlink(V9fsPDU *pdu, V9fsPath *path, V9fsString *buf)
Venkateswararao Jujjuri (JV)86e42d72011-08-08 23:33:48 +053053{
54 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053055 V9fsState *s = pdu->s;
Venkateswararao Jujjuri (JV)86e42d72011-08-08 23:33:48 +053056
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053057 if (v9fs_request_cancelled(pdu)) {
58 return -EINTR;
59 }
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +053060 v9fs_path_read_lock(s);
Venkateswararao Jujjuri (JV)86e42d72011-08-08 23:33:48 +053061 v9fs_co_run_in_worker(
62 {
Chen Gang4fa4ce72014-03-02 01:36:19 +080063 err = __readlink(s, path, buf);
64 if (err < 0) {
Venkateswararao Jujjuri (JV)86e42d72011-08-08 23:33:48 +053065 err = -errno;
66 }
67 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +053068 v9fs_path_unlock(s);
Venkateswararao Jujjuri (JV)86e42d72011-08-08 23:33:48 +053069 return err;
70}
Aneesh Kumar K.V94840ff2011-05-18 16:03:49 -070071
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053072int v9fs_co_statfs(V9fsPDU *pdu, V9fsPath *path, struct statfs *stbuf)
Aneesh Kumar K.V94840ff2011-05-18 16:03:49 -070073{
74 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053075 V9fsState *s = pdu->s;
Aneesh Kumar K.V94840ff2011-05-18 16:03:49 -070076
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053077 if (v9fs_request_cancelled(pdu)) {
78 return -EINTR;
79 }
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +053080 v9fs_path_read_lock(s);
Aneesh Kumar K.V94840ff2011-05-18 16:03:49 -070081 v9fs_co_run_in_worker(
82 {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +053083 err = s->ops->statfs(&s->ctx, path, stbuf);
Aneesh Kumar K.V94840ff2011-05-18 16:03:49 -070084 if (err < 0) {
85 err = -errno;
86 }
87 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +053088 v9fs_path_unlock(s);
Aneesh Kumar K.V94840ff2011-05-18 16:03:49 -070089 return err;
90}
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -070091
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053092int v9fs_co_chmod(V9fsPDU *pdu, V9fsPath *path, mode_t mode)
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -070093{
94 int err;
95 FsCred cred;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053096 V9fsState *s = pdu->s;
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -070097
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +053098 if (v9fs_request_cancelled(pdu)) {
99 return -EINTR;
100 }
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700101 cred_init(&cred);
102 cred.fc_mode = mode;
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530103 v9fs_path_read_lock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700104 v9fs_co_run_in_worker(
105 {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530106 err = s->ops->chmod(&s->ctx, path, &cred);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700107 if (err < 0) {
108 err = -errno;
109 }
110 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530111 v9fs_path_unlock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700112 return err;
113}
114
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530115int v9fs_co_utimensat(V9fsPDU *pdu, V9fsPath *path,
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700116 struct timespec times[2])
117{
118 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530119 V9fsState *s = pdu->s;
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700120
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530121 if (v9fs_request_cancelled(pdu)) {
122 return -EINTR;
123 }
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530124 v9fs_path_read_lock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700125 v9fs_co_run_in_worker(
126 {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530127 err = s->ops->utimensat(&s->ctx, path, times);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700128 if (err < 0) {
129 err = -errno;
130 }
131 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530132 v9fs_path_unlock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700133 return err;
134}
135
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530136int v9fs_co_chown(V9fsPDU *pdu, V9fsPath *path, uid_t uid, gid_t gid)
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700137{
138 int err;
139 FsCred cred;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530140 V9fsState *s = pdu->s;
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700141
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530142 if (v9fs_request_cancelled(pdu)) {
143 return -EINTR;
144 }
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700145 cred_init(&cred);
146 cred.fc_uid = uid;
147 cred.fc_gid = gid;
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530148 v9fs_path_read_lock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700149 v9fs_co_run_in_worker(
150 {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530151 err = s->ops->chown(&s->ctx, path, &cred);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700152 if (err < 0) {
153 err = -errno;
154 }
155 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530156 v9fs_path_unlock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700157 return err;
158}
159
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530160int v9fs_co_truncate(V9fsPDU *pdu, V9fsPath *path, off_t size)
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700161{
162 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530163 V9fsState *s = pdu->s;
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700164
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530165 if (v9fs_request_cancelled(pdu)) {
166 return -EINTR;
167 }
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530168 v9fs_path_read_lock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700169 v9fs_co_run_in_worker(
170 {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530171 err = s->ops->truncate(&s->ctx, path, size);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700172 if (err < 0) {
173 err = -errno;
174 }
175 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530176 v9fs_path_unlock(s);
Aneesh Kumar K.V4011ead2011-05-18 16:04:58 -0700177 return err;
178}
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700179
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530180int v9fs_co_mknod(V9fsPDU *pdu, V9fsFidState *fidp, V9fsString *name, uid_t uid,
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530181 gid_t gid, dev_t dev, mode_t mode, struct stat *stbuf)
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700182{
183 int err;
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530184 V9fsPath path;
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700185 FsCred cred;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530186 V9fsState *s = pdu->s;
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700187
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530188 if (v9fs_request_cancelled(pdu)) {
189 return -EINTR;
190 }
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700191 cred_init(&cred);
192 cred.fc_uid = uid;
193 cred.fc_gid = gid;
194 cred.fc_mode = mode;
195 cred.fc_rdev = dev;
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530196 v9fs_path_read_lock(s);
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700197 v9fs_co_run_in_worker(
198 {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530199 err = s->ops->mknod(&s->ctx, &fidp->path, name->data, &cred);
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700200 if (err < 0) {
201 err = -errno;
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530202 } else {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530203 v9fs_path_init(&path);
204 err = v9fs_name_to_path(s, &fidp->path, name->data, &path);
205 if (!err) {
206 err = s->ops->lstat(&s->ctx, &path, stbuf);
207 if (err < 0) {
208 err = -errno;
209 }
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530210 }
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530211 v9fs_path_free(&path);
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700212 }
213 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530214 v9fs_path_unlock(s);
Aneesh Kumar K.V00ace8c2011-05-18 16:06:38 -0700215 return err;
216}
Venkateswararao Jujjurib4b15372011-08-08 23:48:29 +0530217
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530218/* Only works with path name based fid */
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530219int v9fs_co_remove(V9fsPDU *pdu, V9fsPath *path)
Venkateswararao Jujjurib4b15372011-08-08 23:48:29 +0530220{
221 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530222 V9fsState *s = pdu->s;
Venkateswararao Jujjurib4b15372011-08-08 23:48:29 +0530223
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530224 if (v9fs_request_cancelled(pdu)) {
225 return -EINTR;
226 }
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530227 v9fs_path_read_lock(s);
Venkateswararao Jujjurib4b15372011-08-08 23:48:29 +0530228 v9fs_co_run_in_worker(
229 {
230 err = s->ops->remove(&s->ctx, path->data);
231 if (err < 0) {
232 err = -errno;
233 }
234 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530235 v9fs_path_unlock(s);
Venkateswararao Jujjurib4b15372011-08-08 23:48:29 +0530236 return err;
237}
Aneesh Kumar K.V2a487e02011-05-07 16:09:10 +0530238
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530239int v9fs_co_unlinkat(V9fsPDU *pdu, V9fsPath *path, V9fsString *name, int flags)
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530240{
241 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530242 V9fsState *s = pdu->s;
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530243
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530244 if (v9fs_request_cancelled(pdu)) {
245 return -EINTR;
246 }
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530247 v9fs_path_read_lock(s);
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530248 v9fs_co_run_in_worker(
249 {
250 err = s->ops->unlinkat(&s->ctx, path, name->data, flags);
251 if (err < 0) {
252 err = -errno;
253 }
254 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530255 v9fs_path_unlock(s);
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530256 return err;
257}
258
259/* Only work with path name based fid */
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530260int v9fs_co_rename(V9fsPDU *pdu, V9fsPath *oldpath, V9fsPath *newpath)
Aneesh Kumar K.V2a487e02011-05-07 16:09:10 +0530261{
262 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530263 V9fsState *s = pdu->s;
Aneesh Kumar K.V2a487e02011-05-07 16:09:10 +0530264
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530265 if (v9fs_request_cancelled(pdu)) {
266 return -EINTR;
267 }
Aneesh Kumar K.V2a487e02011-05-07 16:09:10 +0530268 v9fs_co_run_in_worker(
269 {
270 err = s->ops->rename(&s->ctx, oldpath->data, newpath->data);
271 if (err < 0) {
272 err = -errno;
273 }
274 });
275 return err;
276}
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530277
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530278int v9fs_co_renameat(V9fsPDU *pdu, V9fsPath *olddirpath, V9fsString *oldname,
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530279 V9fsPath *newdirpath, V9fsString *newname)
280{
281 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530282 V9fsState *s = pdu->s;
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530283
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530284 if (v9fs_request_cancelled(pdu)) {
285 return -EINTR;
286 }
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530287 v9fs_co_run_in_worker(
288 {
289 err = s->ops->renameat(&s->ctx, olddirpath, oldname->data,
290 newdirpath, newname->data);
291 if (err < 0) {
292 err = -errno;
293 }
294 });
295 return err;
296}
297
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530298int v9fs_co_symlink(V9fsPDU *pdu, V9fsFidState *dfidp, V9fsString *name,
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530299 const char *oldpath, gid_t gid, struct stat *stbuf)
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530300{
301 int err;
302 FsCred cred;
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530303 V9fsPath path;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530304 V9fsState *s = pdu->s;
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530305
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530306 if (v9fs_request_cancelled(pdu)) {
307 return -EINTR;
308 }
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530309 cred_init(&cred);
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530310 cred.fc_uid = dfidp->uid;
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530311 cred.fc_gid = gid;
312 cred.fc_mode = 0777;
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530313 v9fs_path_read_lock(s);
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530314 v9fs_co_run_in_worker(
315 {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530316 err = s->ops->symlink(&s->ctx, oldpath, &dfidp->path,
317 name->data, &cred);
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530318 if (err < 0) {
319 err = -errno;
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530320 } else {
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530321 v9fs_path_init(&path);
322 err = v9fs_name_to_path(s, &dfidp->path, name->data, &path);
323 if (!err) {
324 err = s->ops->lstat(&s->ctx, &path, stbuf);
325 if (err < 0) {
326 err = -errno;
327 }
Aneesh Kumar K.V02cb7f32011-05-24 15:10:56 +0530328 }
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530329 v9fs_path_free(&path);
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530330 }
331 });
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530332 v9fs_path_unlock(s);
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530333 return err;
334}
335
336/*
337 * For path name based fid we don't block. So we can
338 * directly call the fs driver ops.
339 */
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530340int v9fs_co_name_to_path(V9fsPDU *pdu, V9fsPath *dirpath,
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530341 const char *name, V9fsPath *path)
342{
343 int err;
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530344 V9fsState *s = pdu->s;
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530345
Aneesh Kumar K.Vc98f1d42011-10-12 20:59:18 +0530346 if (s->ctx.export_flags & V9FS_PATHNAME_FSCONTEXT) {
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530347 err = s->ops->name_to_path(&s->ctx, dirpath, name, path);
348 if (err < 0) {
349 err = -errno;
350 }
351 } else {
Aneesh Kumar K.Vbccacf62011-08-02 11:36:17 +0530352 if (v9fs_request_cancelled(pdu)) {
353 return -EINTR;
354 }
Aneesh Kumar K.V532decb2011-08-02 11:35:54 +0530355 v9fs_co_run_in_worker(
356 {
357 err = s->ops->name_to_path(&s->ctx, dirpath, name, path);
358 if (err < 0) {
359 err = -errno;
360 }
361 });
Aneesh Kumar K.V2289be12011-09-09 15:14:18 +0530362 }
Venkateswararao Jujjuri02ac7a32011-08-08 23:58:39 +0530363 return err;
364}