]> git.proxmox.com Git - mirror_ubuntu-eoan-kernel.git/blob - drivers/staging/lustre/lustre/obdclass/obd_mount.c
Merge tag 'tegra-for-4.3-cleanup' of git://git.kernel.org/pub/scm/linux/kernel/git...
[mirror_ubuntu-eoan-kernel.git] / drivers / staging / lustre / lustre / obdclass / obd_mount.c
1 /*
2 * GPL HEADER START
3 *
4 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 only,
8 * as published by the Free Software Foundation.
9 *
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * General Public License version 2 for more details (a copy is included
14 * in the LICENSE file that accompanied this code).
15 *
16 * You should have received a copy of the GNU General Public License
17 * version 2 along with this program; If not, see
18 * http://www.sun.com/software/products/lustre/docs/GPLv2.pdf
19 *
20 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
21 * CA 95054 USA or visit www.sun.com if you need additional information or
22 * have any questions.
23 *
24 * GPL HEADER END
25 */
26 /*
27 * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
28 * Use is subject to license terms.
29 *
30 * Copyright (c) 2011, 2012, Intel Corporation.
31 */
32 /*
33 * This file is part of Lustre, http://www.lustre.org/
34 * Lustre is a trademark of Sun Microsystems, Inc.
35 *
36 * lustre/obdclass/obd_mount.c
37 *
38 * Client mount routines
39 *
40 * Author: Nathan Rutman <nathan@clusterfs.com>
41 */
42
43
44 #define DEBUG_SUBSYSTEM S_CLASS
45 #define D_MOUNT (D_SUPER|D_CONFIG/*|D_WARNING */)
46 #define PRINT_CMD CDEBUG
47
48 #include "../include/obd.h"
49 #include "../include/linux/lustre_compat25.h"
50 #include "../include/obd_class.h"
51 #include "../include/lustre/lustre_user.h"
52 #include "../include/lustre_log.h"
53 #include "../include/lustre_disk.h"
54 #include "../include/lustre_param.h"
55
56 static int (*client_fill_super)(struct super_block *sb,
57 struct vfsmount *mnt);
58
59 static void (*kill_super_cb)(struct super_block *sb);
60
61 /**************** config llog ********************/
62
63 /** Get a config log from the MGS and process it.
64 * This func is called for both clients and servers.
65 * Continue to process new statements appended to the logs
66 * (whenever the config lock is revoked) until lustre_end_log
67 * is called.
68 * @param sb The superblock is used by the MGC to write to the local copy of
69 * the config log
70 * @param logname The name of the llog to replicate from the MGS
71 * @param cfg Since the same mgc may be used to follow multiple config logs
72 * (e.g. ost1, ost2, client), the config_llog_instance keeps the state for
73 * this log, and is added to the mgc's list of logs to follow.
74 */
75 int lustre_process_log(struct super_block *sb, char *logname,
76 struct config_llog_instance *cfg)
77 {
78 struct lustre_cfg *lcfg;
79 struct lustre_cfg_bufs *bufs;
80 struct lustre_sb_info *lsi = s2lsi(sb);
81 struct obd_device *mgc = lsi->lsi_mgc;
82 int rc;
83
84 LASSERT(mgc);
85 LASSERT(cfg);
86
87 bufs = kzalloc(sizeof(*bufs), GFP_NOFS);
88 if (bufs == NULL)
89 return -ENOMEM;
90
91 /* mgc_process_config */
92 lustre_cfg_bufs_reset(bufs, mgc->obd_name);
93 lustre_cfg_bufs_set_string(bufs, 1, logname);
94 lustre_cfg_bufs_set(bufs, 2, cfg, sizeof(*cfg));
95 lustre_cfg_bufs_set(bufs, 3, &sb, sizeof(sb));
96 lcfg = lustre_cfg_new(LCFG_LOG_START, bufs);
97 rc = obd_process_config(mgc, sizeof(*lcfg), lcfg);
98 lustre_cfg_free(lcfg);
99
100 kfree(bufs);
101
102 if (rc == -EINVAL)
103 LCONSOLE_ERROR_MSG(0x15b, "%s: The configuration from log '%s' failed from the MGS (%d). Make sure this client and the MGS are running compatible versions of Lustre.\n",
104 mgc->obd_name, logname, rc);
105
106 if (rc)
107 LCONSOLE_ERROR_MSG(0x15c, "%s: The configuration from log '%s' failed (%d). This may be the result of communication errors between this node and the MGS, a bad configuration, or other errors. See the syslog for more information.\n",
108 mgc->obd_name, logname,
109 rc);
110
111 /* class_obd_list(); */
112 return rc;
113 }
114 EXPORT_SYMBOL(lustre_process_log);
115
116 /* Stop watching this config log for updates */
117 int lustre_end_log(struct super_block *sb, char *logname,
118 struct config_llog_instance *cfg)
119 {
120 struct lustre_cfg *lcfg;
121 struct lustre_cfg_bufs bufs;
122 struct lustre_sb_info *lsi = s2lsi(sb);
123 struct obd_device *mgc = lsi->lsi_mgc;
124 int rc;
125
126 if (!mgc)
127 return -ENOENT;
128
129 /* mgc_process_config */
130 lustre_cfg_bufs_reset(&bufs, mgc->obd_name);
131 lustre_cfg_bufs_set_string(&bufs, 1, logname);
132 if (cfg)
133 lustre_cfg_bufs_set(&bufs, 2, cfg, sizeof(*cfg));
134 lcfg = lustre_cfg_new(LCFG_LOG_END, &bufs);
135 rc = obd_process_config(mgc, sizeof(*lcfg), lcfg);
136 lustre_cfg_free(lcfg);
137 return rc;
138 }
139 EXPORT_SYMBOL(lustre_end_log);
140
141 /**************** obd start *******************/
142
143 /** lustre_cfg_bufs are a holdover from 1.4; we can still set these up from
144 * lctl (and do for echo cli/srv.
145 */
146 int do_lcfg(char *cfgname, lnet_nid_t nid, int cmd,
147 char *s1, char *s2, char *s3, char *s4)
148 {
149 struct lustre_cfg_bufs bufs;
150 struct lustre_cfg *lcfg = NULL;
151 int rc;
152
153 CDEBUG(D_TRACE, "lcfg %s %#x %s %s %s %s\n", cfgname,
154 cmd, s1, s2, s3, s4);
155
156 lustre_cfg_bufs_reset(&bufs, cfgname);
157 if (s1)
158 lustre_cfg_bufs_set_string(&bufs, 1, s1);
159 if (s2)
160 lustre_cfg_bufs_set_string(&bufs, 2, s2);
161 if (s3)
162 lustre_cfg_bufs_set_string(&bufs, 3, s3);
163 if (s4)
164 lustre_cfg_bufs_set_string(&bufs, 4, s4);
165
166 lcfg = lustre_cfg_new(cmd, &bufs);
167 lcfg->lcfg_nid = nid;
168 rc = class_process_config(lcfg);
169 lustre_cfg_free(lcfg);
170 return rc;
171 }
172 EXPORT_SYMBOL(do_lcfg);
173
174 /** Call class_attach and class_setup. These methods in turn call
175 * obd type-specific methods.
176 */
177 int lustre_start_simple(char *obdname, char *type, char *uuid,
178 char *s1, char *s2, char *s3, char *s4)
179 {
180 int rc;
181 CDEBUG(D_MOUNT, "Starting obd %s (typ=%s)\n", obdname, type);
182
183 rc = do_lcfg(obdname, 0, LCFG_ATTACH, type, uuid, NULL, NULL);
184 if (rc) {
185 CERROR("%s attach error %d\n", obdname, rc);
186 return rc;
187 }
188 rc = do_lcfg(obdname, 0, LCFG_SETUP, s1, s2, s3, s4);
189 if (rc) {
190 CERROR("%s setup error %d\n", obdname, rc);
191 do_lcfg(obdname, 0, LCFG_DETACH, NULL, NULL, NULL, NULL);
192 }
193 return rc;
194 }
195
196 DEFINE_MUTEX(mgc_start_lock);
197
198 /** Set up a mgc obd to process startup logs
199 *
200 * \param sb [in] super block of the mgc obd
201 *
202 * \retval 0 success, otherwise error code
203 */
204 int lustre_start_mgc(struct super_block *sb)
205 {
206 struct obd_connect_data *data = NULL;
207 struct lustre_sb_info *lsi = s2lsi(sb);
208 struct obd_device *obd;
209 struct obd_export *exp;
210 struct obd_uuid *uuid;
211 class_uuid_t uuidc;
212 lnet_nid_t nid;
213 char *mgcname = NULL, *niduuid = NULL, *mgssec = NULL;
214 char *ptr;
215 int rc = 0, i = 0, j, len;
216
217 LASSERT(lsi->lsi_lmd);
218
219 /* Find the first non-lo MGS nid for our MGC name */
220 if (IS_SERVER(lsi)) {
221 /* mount -o mgsnode=nid */
222 ptr = lsi->lsi_lmd->lmd_mgs;
223 if (lsi->lsi_lmd->lmd_mgs &&
224 (class_parse_nid(lsi->lsi_lmd->lmd_mgs, &nid, &ptr) == 0)) {
225 i++;
226 } else if (IS_MGS(lsi)) {
227 lnet_process_id_t id;
228 while ((rc = LNetGetId(i++, &id)) != -ENOENT) {
229 if (LNET_NETTYP(LNET_NIDNET(id.nid)) == LOLND)
230 continue;
231 nid = id.nid;
232 i++;
233 break;
234 }
235 }
236 } else { /* client */
237 /* Use nids from mount line: uml1,1@elan:uml2,2@elan:/lustre */
238 ptr = lsi->lsi_lmd->lmd_dev;
239 if (class_parse_nid(ptr, &nid, &ptr) == 0)
240 i++;
241 }
242 if (i == 0) {
243 CERROR("No valid MGS nids found.\n");
244 return -EINVAL;
245 }
246
247 mutex_lock(&mgc_start_lock);
248
249 len = strlen(LUSTRE_MGC_OBDNAME) + strlen(libcfs_nid2str(nid)) + 1;
250 mgcname = kzalloc(len, GFP_NOFS);
251 niduuid = kzalloc(len + 2, GFP_NOFS);
252 if (!mgcname || !niduuid) {
253 rc = -ENOMEM;
254 goto out_free;
255 }
256 sprintf(mgcname, "%s%s", LUSTRE_MGC_OBDNAME, libcfs_nid2str(nid));
257
258 mgssec = lsi->lsi_lmd->lmd_mgssec ? lsi->lsi_lmd->lmd_mgssec : "";
259
260 data = kzalloc(sizeof(*data), GFP_NOFS);
261 if (data == NULL) {
262 rc = -ENOMEM;
263 goto out_free;
264 }
265
266 obd = class_name2obd(mgcname);
267 if (obd && !obd->obd_stopping) {
268 int recov_bk;
269
270 rc = obd_set_info_async(NULL, obd->obd_self_export,
271 strlen(KEY_MGSSEC), KEY_MGSSEC,
272 strlen(mgssec), mgssec, NULL);
273 if (rc)
274 goto out_free;
275
276 /* Re-using an existing MGC */
277 atomic_inc(&obd->u.cli.cl_mgc_refcount);
278
279 /* IR compatibility check, only for clients */
280 if (lmd_is_client(lsi->lsi_lmd)) {
281 int has_ir;
282 int vallen = sizeof(*data);
283 __u32 *flags = &lsi->lsi_lmd->lmd_flags;
284
285 rc = obd_get_info(NULL, obd->obd_self_export,
286 strlen(KEY_CONN_DATA), KEY_CONN_DATA,
287 &vallen, data, NULL);
288 LASSERT(rc == 0);
289 has_ir = OCD_HAS_FLAG(data, IMP_RECOV);
290 if (has_ir ^ !(*flags & LMD_FLG_NOIR)) {
291 /* LMD_FLG_NOIR is for test purpose only */
292 LCONSOLE_WARN(
293 "Trying to mount a client with IR setting not compatible with current mgc. Force to use current mgc setting that is IR %s.\n",
294 has_ir ? "enabled" : "disabled");
295 if (has_ir)
296 *flags &= ~LMD_FLG_NOIR;
297 else
298 *flags |= LMD_FLG_NOIR;
299 }
300 }
301
302 recov_bk = 0;
303 /* If we are restarting the MGS, don't try to keep the MGC's
304 old connection, or registration will fail. */
305 if (IS_MGS(lsi)) {
306 CDEBUG(D_MOUNT, "New MGS with live MGC\n");
307 recov_bk = 1;
308 }
309
310 /* Try all connections, but only once (again).
311 We don't want to block another target from starting
312 (using its local copy of the log), but we do want to connect
313 if at all possible. */
314 recov_bk++;
315 CDEBUG(D_MOUNT, "%s: Set MGC reconnect %d\n", mgcname,
316 recov_bk);
317 rc = obd_set_info_async(NULL, obd->obd_self_export,
318 sizeof(KEY_INIT_RECOV_BACKUP),
319 KEY_INIT_RECOV_BACKUP,
320 sizeof(recov_bk), &recov_bk, NULL);
321 rc = 0;
322 goto out;
323 }
324
325 CDEBUG(D_MOUNT, "Start MGC '%s'\n", mgcname);
326
327 /* Add the primary nids for the MGS */
328 i = 0;
329 sprintf(niduuid, "%s_%x", mgcname, i);
330 if (IS_SERVER(lsi)) {
331 ptr = lsi->lsi_lmd->lmd_mgs;
332 if (IS_MGS(lsi)) {
333 /* Use local nids (including LO) */
334 lnet_process_id_t id;
335 while ((rc = LNetGetId(i++, &id)) != -ENOENT) {
336 rc = do_lcfg(mgcname, id.nid,
337 LCFG_ADD_UUID, niduuid,
338 NULL, NULL, NULL);
339 }
340 } else {
341 /* Use mgsnode= nids */
342 /* mount -o mgsnode=nid */
343 if (lsi->lsi_lmd->lmd_mgs) {
344 ptr = lsi->lsi_lmd->lmd_mgs;
345 } else if (class_find_param(ptr, PARAM_MGSNODE,
346 &ptr) != 0) {
347 CERROR("No MGS nids given.\n");
348 rc = -EINVAL;
349 goto out_free;
350 }
351 while (class_parse_nid(ptr, &nid, &ptr) == 0) {
352 rc = do_lcfg(mgcname, nid,
353 LCFG_ADD_UUID, niduuid,
354 NULL, NULL, NULL);
355 i++;
356 }
357 }
358 } else { /* client */
359 /* Use nids from mount line: uml1,1@elan:uml2,2@elan:/lustre */
360 ptr = lsi->lsi_lmd->lmd_dev;
361 while (class_parse_nid(ptr, &nid, &ptr) == 0) {
362 rc = do_lcfg(mgcname, nid,
363 LCFG_ADD_UUID, niduuid, NULL, NULL, NULL);
364 i++;
365 /* Stop at the first failover nid */
366 if (*ptr == ':')
367 break;
368 }
369 }
370 if (i == 0) {
371 CERROR("No valid MGS nids found.\n");
372 rc = -EINVAL;
373 goto out_free;
374 }
375 lsi->lsi_lmd->lmd_mgs_failnodes = 1;
376
377 /* Random uuid for MGC allows easier reconnects */
378 uuid = kzalloc(sizeof(*uuid), GFP_NOFS);
379 if (!uuid) {
380 rc = -ENOMEM;
381 goto out_free;
382 }
383
384 ll_generate_random_uuid(uuidc);
385 class_uuid_unparse(uuidc, uuid);
386
387 /* Start the MGC */
388 rc = lustre_start_simple(mgcname, LUSTRE_MGC_NAME,
389 (char *)uuid->uuid, LUSTRE_MGS_OBDNAME,
390 niduuid, NULL, NULL);
391 kfree(uuid);
392 if (rc)
393 goto out_free;
394
395 /* Add any failover MGS nids */
396 i = 1;
397 while (ptr && ((*ptr == ':' ||
398 class_find_param(ptr, PARAM_MGSNODE, &ptr) == 0))) {
399 /* New failover node */
400 sprintf(niduuid, "%s_%x", mgcname, i);
401 j = 0;
402 while (class_parse_nid_quiet(ptr, &nid, &ptr) == 0) {
403 j++;
404 rc = do_lcfg(mgcname, nid,
405 LCFG_ADD_UUID, niduuid, NULL, NULL, NULL);
406 if (*ptr == ':')
407 break;
408 }
409 if (j > 0) {
410 rc = do_lcfg(mgcname, 0, LCFG_ADD_CONN,
411 niduuid, NULL, NULL, NULL);
412 i++;
413 } else {
414 /* at ":/fsname" */
415 break;
416 }
417 }
418 lsi->lsi_lmd->lmd_mgs_failnodes = i;
419
420 obd = class_name2obd(mgcname);
421 if (!obd) {
422 CERROR("Can't find mgcobd %s\n", mgcname);
423 rc = -ENOTCONN;
424 goto out_free;
425 }
426
427 rc = obd_set_info_async(NULL, obd->obd_self_export,
428 strlen(KEY_MGSSEC), KEY_MGSSEC,
429 strlen(mgssec), mgssec, NULL);
430 if (rc)
431 goto out_free;
432
433 /* Keep a refcount of servers/clients who started with "mount",
434 so we know when we can get rid of the mgc. */
435 atomic_set(&obd->u.cli.cl_mgc_refcount, 1);
436
437 /* We connect to the MGS at setup, and don't disconnect until cleanup */
438 data->ocd_connect_flags = OBD_CONNECT_VERSION | OBD_CONNECT_AT |
439 OBD_CONNECT_FULL20 | OBD_CONNECT_IMP_RECOV |
440 OBD_CONNECT_LVB_TYPE;
441
442 #if LUSTRE_VERSION_CODE < OBD_OCD_VERSION(3, 2, 50, 0)
443 data->ocd_connect_flags |= OBD_CONNECT_MNE_SWAB;
444 #else
445 #warning "LU-1644: Remove old OBD_CONNECT_MNE_SWAB fixup and imp_need_mne_swab"
446 #endif
447
448 if (lmd_is_client(lsi->lsi_lmd) &&
449 lsi->lsi_lmd->lmd_flags & LMD_FLG_NOIR)
450 data->ocd_connect_flags &= ~OBD_CONNECT_IMP_RECOV;
451 data->ocd_version = LUSTRE_VERSION_CODE;
452 rc = obd_connect(NULL, &exp, obd, &(obd->obd_uuid), data, NULL);
453 if (rc) {
454 CERROR("connect failed %d\n", rc);
455 goto out;
456 }
457
458 obd->u.cli.cl_mgc_mgsexp = exp;
459
460 out:
461 /* Keep the mgc info in the sb. Note that many lsi's can point
462 to the same mgc.*/
463 lsi->lsi_mgc = obd;
464 out_free:
465 mutex_unlock(&mgc_start_lock);
466
467 kfree(data);
468 kfree(mgcname);
469 kfree(niduuid);
470 return rc;
471 }
472
473 static int lustre_stop_mgc(struct super_block *sb)
474 {
475 struct lustre_sb_info *lsi = s2lsi(sb);
476 struct obd_device *obd;
477 char *niduuid = NULL, *ptr = NULL;
478 int i, rc = 0, len = 0;
479
480 if (!lsi)
481 return -ENOENT;
482 obd = lsi->lsi_mgc;
483 if (!obd)
484 return -ENOENT;
485 lsi->lsi_mgc = NULL;
486
487 mutex_lock(&mgc_start_lock);
488 LASSERT(atomic_read(&obd->u.cli.cl_mgc_refcount) > 0);
489 if (!atomic_dec_and_test(&obd->u.cli.cl_mgc_refcount)) {
490 /* This is not fatal, every client that stops
491 will call in here. */
492 CDEBUG(D_MOUNT, "mgc still has %d references.\n",
493 atomic_read(&obd->u.cli.cl_mgc_refcount));
494 rc = -EBUSY;
495 goto out;
496 }
497
498 /* The MGC has no recoverable data in any case.
499 * force shutdown set in umount_begin */
500 obd->obd_no_recov = 1;
501
502 if (obd->u.cli.cl_mgc_mgsexp) {
503 /* An error is not fatal, if we are unable to send the
504 disconnect mgs ping evictor cleans up the export */
505 rc = obd_disconnect(obd->u.cli.cl_mgc_mgsexp);
506 if (rc)
507 CDEBUG(D_MOUNT, "disconnect failed %d\n", rc);
508 }
509
510 /* Save the obdname for cleaning the nid uuids, which are
511 obdname_XX */
512 len = strlen(obd->obd_name) + 6;
513 niduuid = kzalloc(len, GFP_NOFS);
514 if (niduuid) {
515 strcpy(niduuid, obd->obd_name);
516 ptr = niduuid + strlen(niduuid);
517 }
518
519 rc = class_manual_cleanup(obd);
520 if (rc)
521 goto out;
522
523 /* Clean the nid uuids */
524 if (!niduuid) {
525 rc = -ENOMEM;
526 goto out;
527 }
528
529 for (i = 0; i < lsi->lsi_lmd->lmd_mgs_failnodes; i++) {
530 sprintf(ptr, "_%x", i);
531 rc = do_lcfg(LUSTRE_MGC_OBDNAME, 0, LCFG_DEL_UUID,
532 niduuid, NULL, NULL, NULL);
533 if (rc)
534 CERROR("del MDC UUID %s failed: rc = %d\n",
535 niduuid, rc);
536 }
537 out:
538 kfree(niduuid);
539
540 /* class_import_put will get rid of the additional connections */
541 mutex_unlock(&mgc_start_lock);
542 return rc;
543 }
544
545 /***************** lustre superblock **************/
546
547 struct lustre_sb_info *lustre_init_lsi(struct super_block *sb)
548 {
549 struct lustre_sb_info *lsi;
550
551 lsi = kzalloc(sizeof(*lsi), GFP_NOFS);
552 if (!lsi)
553 return NULL;
554 lsi->lsi_lmd = kzalloc(sizeof(*lsi->lsi_lmd), GFP_NOFS);
555 if (!lsi->lsi_lmd) {
556 kfree(lsi);
557 return NULL;
558 }
559
560 lsi->lsi_lmd->lmd_exclude_count = 0;
561 lsi->lsi_lmd->lmd_recovery_time_soft = 0;
562 lsi->lsi_lmd->lmd_recovery_time_hard = 0;
563 s2lsi_nocast(sb) = lsi;
564 /* we take 1 extra ref for our setup */
565 atomic_set(&lsi->lsi_mounts, 1);
566
567 /* Default umount style */
568 lsi->lsi_flags = LSI_UMOUNT_FAILOVER;
569
570 return lsi;
571 }
572
573 static int lustre_free_lsi(struct super_block *sb)
574 {
575 struct lustre_sb_info *lsi = s2lsi(sb);
576
577 LASSERT(lsi != NULL);
578 CDEBUG(D_MOUNT, "Freeing lsi %p\n", lsi);
579
580 /* someone didn't call server_put_mount. */
581 LASSERT(atomic_read(&lsi->lsi_mounts) == 0);
582
583 if (lsi->lsi_lmd != NULL) {
584 kfree(lsi->lsi_lmd->lmd_dev);
585 kfree(lsi->lsi_lmd->lmd_profile);
586 kfree(lsi->lsi_lmd->lmd_mgssec);
587 kfree(lsi->lsi_lmd->lmd_opts);
588 if (lsi->lsi_lmd->lmd_exclude_count)
589 kfree(lsi->lsi_lmd->lmd_exclude);
590 kfree(lsi->lsi_lmd->lmd_mgs);
591 kfree(lsi->lsi_lmd->lmd_osd_type);
592 kfree(lsi->lsi_lmd->lmd_params);
593
594 kfree(lsi->lsi_lmd);
595 }
596
597 LASSERT(lsi->lsi_llsbi == NULL);
598 kfree(lsi);
599 s2lsi_nocast(sb) = NULL;
600
601 return 0;
602 }
603
604 /* The lsi has one reference for every server that is using the disk -
605 e.g. MDT, MGS, and potentially MGC */
606 int lustre_put_lsi(struct super_block *sb)
607 {
608 struct lustre_sb_info *lsi = s2lsi(sb);
609
610 LASSERT(lsi != NULL);
611
612 CDEBUG(D_MOUNT, "put %p %d\n", sb, atomic_read(&lsi->lsi_mounts));
613 if (atomic_dec_and_test(&lsi->lsi_mounts)) {
614 if (IS_SERVER(lsi) && lsi->lsi_osd_exp) {
615 lu_device_put(&lsi->lsi_dt_dev->dd_lu_dev);
616 lsi->lsi_osd_exp->exp_obd->obd_lvfs_ctxt.dt = NULL;
617 lsi->lsi_dt_dev = NULL;
618 obd_disconnect(lsi->lsi_osd_exp);
619 /* wait till OSD is gone */
620 obd_zombie_barrier();
621 }
622 lustre_free_lsi(sb);
623 return 1;
624 }
625 return 0;
626 }
627
628 /*** SERVER NAME ***
629 * <FSNAME><SEPARATOR><TYPE><INDEX>
630 * FSNAME is between 1 and 8 characters (inclusive).
631 * Excluded characters are '/' and ':'
632 * SEPARATOR is either ':' or '-'
633 * TYPE: "OST", "MDT", etc.
634 * INDEX: Hex representation of the index
635 */
636
637 /** Get the fsname ("lustre") from the server name ("lustre-OST003F").
638 * @param [in] svname server name including type and index
639 * @param [out] fsname Buffer to copy filesystem name prefix into.
640 * Must have at least 'strlen(fsname) + 1' chars.
641 * @param [out] endptr if endptr isn't NULL it is set to end of fsname
642 * rc < 0 on error
643 */
644 int server_name2fsname(const char *svname, char *fsname, const char **endptr)
645 {
646 const char *dash;
647
648 dash = svname + strnlen(svname, 8); /* max fsname length is 8 */
649 for (; dash > svname && *dash != '-' && *dash != ':'; dash--)
650 ;
651 if (dash == svname)
652 return -EINVAL;
653
654 if (fsname != NULL) {
655 strncpy(fsname, svname, dash - svname);
656 fsname[dash - svname] = '\0';
657 }
658
659 if (endptr != NULL)
660 *endptr = dash;
661
662 return 0;
663 }
664 EXPORT_SYMBOL(server_name2fsname);
665
666 /**
667 * Get service name (svname) from string
668 * rc < 0 on error
669 * if endptr isn't NULL it is set to end of fsname *
670 */
671 int server_name2svname(const char *label, char *svname, const char **endptr,
672 size_t svsize)
673 {
674 int rc;
675 const char *dash;
676
677 /* We use server_name2fsname() just for parsing */
678 rc = server_name2fsname(label, NULL, &dash);
679 if (rc != 0)
680 return rc;
681
682 if (endptr != NULL)
683 *endptr = dash;
684
685 if (strlcpy(svname, dash + 1, svsize) >= svsize)
686 return -E2BIG;
687
688 return 0;
689 }
690 EXPORT_SYMBOL(server_name2svname);
691
692
693 /* Get the index from the obd name.
694 rc = server type, or
695 rc < 0 on error
696 if endptr isn't NULL it is set to end of name */
697 int server_name2index(const char *svname, __u32 *idx, const char **endptr)
698 {
699 unsigned long index;
700 int rc;
701 const char *dash;
702
703 /* We use server_name2fsname() just for parsing */
704 rc = server_name2fsname(svname, NULL, &dash);
705 if (rc != 0)
706 return rc;
707
708 dash++;
709
710 if (strncmp(dash, "MDT", 3) == 0)
711 rc = LDD_F_SV_TYPE_MDT;
712 else if (strncmp(dash, "OST", 3) == 0)
713 rc = LDD_F_SV_TYPE_OST;
714 else
715 return -EINVAL;
716
717 dash += 3;
718
719 if (strncmp(dash, "all", 3) == 0) {
720 if (endptr != NULL)
721 *endptr = dash + 3;
722 return rc | LDD_F_SV_ALL;
723 }
724
725 index = simple_strtoul(dash, (char **)endptr, 16);
726 if (idx != NULL)
727 *idx = index;
728
729 /* Account for -mdc after index that is possible when specifying mdt */
730 if (endptr != NULL && strncmp(LUSTRE_MDC_NAME, *endptr + 1,
731 sizeof(LUSTRE_MDC_NAME)-1) == 0)
732 *endptr += sizeof(LUSTRE_MDC_NAME);
733
734 return rc;
735 }
736 EXPORT_SYMBOL(server_name2index);
737
738 /*************** mount common between server and client ***************/
739
740 /* Common umount */
741 int lustre_common_put_super(struct super_block *sb)
742 {
743 int rc;
744
745 CDEBUG(D_MOUNT, "dropping sb %p\n", sb);
746
747 /* Drop a ref to the MGC */
748 rc = lustre_stop_mgc(sb);
749 if (rc && (rc != -ENOENT)) {
750 if (rc != -EBUSY) {
751 CERROR("Can't stop MGC: %d\n", rc);
752 return rc;
753 }
754 /* BUSY just means that there's some other obd that
755 needs the mgc. Let him clean it up. */
756 CDEBUG(D_MOUNT, "MGC still in use\n");
757 }
758 /* Drop a ref to the mounted disk */
759 lustre_put_lsi(sb);
760 lu_types_stop();
761 return rc;
762 }
763 EXPORT_SYMBOL(lustre_common_put_super);
764
765 static void lmd_print(struct lustre_mount_data *lmd)
766 {
767 int i;
768
769 PRINT_CMD(D_MOUNT, " mount data:\n");
770 if (lmd_is_client(lmd))
771 PRINT_CMD(D_MOUNT, "profile: %s\n", lmd->lmd_profile);
772 PRINT_CMD(D_MOUNT, "device: %s\n", lmd->lmd_dev);
773 PRINT_CMD(D_MOUNT, "flags: %x\n", lmd->lmd_flags);
774
775 if (lmd->lmd_opts)
776 PRINT_CMD(D_MOUNT, "options: %s\n", lmd->lmd_opts);
777
778 if (lmd->lmd_recovery_time_soft)
779 PRINT_CMD(D_MOUNT, "recovery time soft: %d\n",
780 lmd->lmd_recovery_time_soft);
781
782 if (lmd->lmd_recovery_time_hard)
783 PRINT_CMD(D_MOUNT, "recovery time hard: %d\n",
784 lmd->lmd_recovery_time_hard);
785
786 for (i = 0; i < lmd->lmd_exclude_count; i++) {
787 PRINT_CMD(D_MOUNT, "exclude %d: OST%04x\n", i,
788 lmd->lmd_exclude[i]);
789 }
790 }
791
792 /* Is this server on the exclusion list */
793 int lustre_check_exclusion(struct super_block *sb, char *svname)
794 {
795 struct lustre_sb_info *lsi = s2lsi(sb);
796 struct lustre_mount_data *lmd = lsi->lsi_lmd;
797 __u32 index;
798 int i, rc;
799
800 rc = server_name2index(svname, &index, NULL);
801 if (rc != LDD_F_SV_TYPE_OST)
802 /* Only exclude OSTs */
803 return 0;
804
805 CDEBUG(D_MOUNT, "Check exclusion %s (%d) in %d of %s\n", svname,
806 index, lmd->lmd_exclude_count, lmd->lmd_dev);
807
808 for (i = 0; i < lmd->lmd_exclude_count; i++) {
809 if (index == lmd->lmd_exclude[i]) {
810 CWARN("Excluding %s (on exclusion list)\n", svname);
811 return 1;
812 }
813 }
814 return 0;
815 }
816
817 /* mount -v -o exclude=lustre-OST0001:lustre-OST0002 -t lustre ... */
818 static int lmd_make_exclusion(struct lustre_mount_data *lmd, const char *ptr)
819 {
820 const char *s1 = ptr, *s2;
821 __u32 index, *exclude_list;
822 int rc = 0, devmax;
823
824 /* The shortest an ost name can be is 8 chars: -OST0000.
825 We don't actually know the fsname at this time, so in fact
826 a user could specify any fsname. */
827 devmax = strlen(ptr) / 8 + 1;
828
829 /* temp storage until we figure out how many we have */
830 exclude_list = kcalloc(devmax, sizeof(index), GFP_NOFS);
831 if (!exclude_list)
832 return -ENOMEM;
833
834 /* we enter this fn pointing at the '=' */
835 while (*s1 && *s1 != ' ' && *s1 != ',') {
836 s1++;
837 rc = server_name2index(s1, &index, &s2);
838 if (rc < 0) {
839 CERROR("Can't parse server name '%s': rc = %d\n",
840 s1, rc);
841 break;
842 }
843 if (rc == LDD_F_SV_TYPE_OST)
844 exclude_list[lmd->lmd_exclude_count++] = index;
845 else
846 CDEBUG(D_MOUNT, "ignoring exclude %.*s: type = %#x\n",
847 (uint)(s2-s1), s1, rc);
848 s1 = s2;
849 /* now we are pointing at ':' (next exclude)
850 or ',' (end of excludes) */
851 if (lmd->lmd_exclude_count >= devmax)
852 break;
853 }
854 if (rc >= 0) /* non-err */
855 rc = 0;
856
857 if (lmd->lmd_exclude_count) {
858 /* permanent, freed in lustre_free_lsi */
859 lmd->lmd_exclude = kcalloc(lmd->lmd_exclude_count,
860 sizeof(index), GFP_NOFS);
861 if (lmd->lmd_exclude) {
862 memcpy(lmd->lmd_exclude, exclude_list,
863 sizeof(index) * lmd->lmd_exclude_count);
864 } else {
865 rc = -ENOMEM;
866 lmd->lmd_exclude_count = 0;
867 }
868 }
869 kfree(exclude_list);
870 return rc;
871 }
872
873 static int lmd_parse_mgssec(struct lustre_mount_data *lmd, char *ptr)
874 {
875 char *tail;
876 int length;
877
878 kfree(lmd->lmd_mgssec);
879 lmd->lmd_mgssec = NULL;
880
881 tail = strchr(ptr, ',');
882 if (tail == NULL)
883 length = strlen(ptr);
884 else
885 length = tail - ptr;
886
887 lmd->lmd_mgssec = kzalloc(length + 1, GFP_NOFS);
888 if (lmd->lmd_mgssec == NULL)
889 return -ENOMEM;
890
891 memcpy(lmd->lmd_mgssec, ptr, length);
892 lmd->lmd_mgssec[length] = '\0';
893 return 0;
894 }
895
896 static int lmd_parse_string(char **handle, char *ptr)
897 {
898 char *tail;
899 int length;
900
901 if ((handle == NULL) || (ptr == NULL))
902 return -EINVAL;
903
904 kfree(*handle);
905 *handle = NULL;
906
907 tail = strchr(ptr, ',');
908 if (tail == NULL)
909 length = strlen(ptr);
910 else
911 length = tail - ptr;
912
913 *handle = kzalloc(length + 1, GFP_NOFS);
914 if (*handle == NULL)
915 return -ENOMEM;
916
917 memcpy(*handle, ptr, length);
918 (*handle)[length] = '\0';
919
920 return 0;
921 }
922
923 /* Collect multiple values for mgsnid specifiers */
924 static int lmd_parse_mgs(struct lustre_mount_data *lmd, char **ptr)
925 {
926 lnet_nid_t nid;
927 char *tail = *ptr;
928 char *mgsnid;
929 int length;
930 int oldlen = 0;
931
932 /* Find end of nidlist */
933 while (class_parse_nid_quiet(tail, &nid, &tail) == 0) {}
934 length = tail - *ptr;
935 if (length == 0) {
936 LCONSOLE_ERROR_MSG(0x159, "Can't parse NID '%s'\n", *ptr);
937 return -EINVAL;
938 }
939
940 if (lmd->lmd_mgs != NULL)
941 oldlen = strlen(lmd->lmd_mgs) + 1;
942
943 mgsnid = kzalloc(oldlen + length + 1, GFP_NOFS);
944 if (mgsnid == NULL)
945 return -ENOMEM;
946
947 if (lmd->lmd_mgs != NULL) {
948 /* Multiple mgsnid= are taken to mean failover locations */
949 memcpy(mgsnid, lmd->lmd_mgs, oldlen);
950 mgsnid[oldlen - 1] = ':';
951 kfree(lmd->lmd_mgs);
952 }
953 memcpy(mgsnid + oldlen, *ptr, length);
954 mgsnid[oldlen + length] = '\0';
955 lmd->lmd_mgs = mgsnid;
956 *ptr = tail;
957
958 return 0;
959 }
960
961 /** Parse mount line options
962 * e.g. mount -v -t lustre -o abort_recov uml1:uml2:/lustre-client /mnt/lustre
963 * dev is passed as device=uml1:/lustre by mount.lustre
964 */
965 static int lmd_parse(char *options, struct lustre_mount_data *lmd)
966 {
967 char *s1, *s2, *devname = NULL;
968 struct lustre_mount_data *raw = (struct lustre_mount_data *)options;
969 int rc = 0;
970
971 LASSERT(lmd);
972 if (!options) {
973 LCONSOLE_ERROR_MSG(0x162, "Missing mount data: check that /sbin/mount.lustre is installed.\n");
974 return -EINVAL;
975 }
976
977 /* Options should be a string - try to detect old lmd data */
978 if ((raw->lmd_magic & 0xffffff00) == (LMD_MAGIC & 0xffffff00)) {
979 LCONSOLE_ERROR_MSG(0x163, "You're using an old version of /sbin/mount.lustre. Please install version %s\n",
980 LUSTRE_VERSION_STRING);
981 return -EINVAL;
982 }
983 lmd->lmd_magic = LMD_MAGIC;
984
985 lmd->lmd_params = kzalloc(4096, GFP_NOFS);
986 if (lmd->lmd_params == NULL)
987 return -ENOMEM;
988 lmd->lmd_params[0] = '\0';
989
990 /* Set default flags here */
991
992 s1 = options;
993 while (*s1) {
994 int clear = 0;
995 int time_min = OBD_RECOVERY_TIME_MIN;
996
997 /* Skip whitespace and extra commas */
998 while (*s1 == ' ' || *s1 == ',')
999 s1++;
1000
1001 /* Client options are parsed in ll_options: eg. flock,
1002 user_xattr, acl */
1003
1004 /* Parse non-ldiskfs options here. Rather than modifying
1005 ldiskfs, we just zero these out here */
1006 if (strncmp(s1, "abort_recov", 11) == 0) {
1007 lmd->lmd_flags |= LMD_FLG_ABORT_RECOV;
1008 clear++;
1009 } else if (strncmp(s1, "recovery_time_soft=", 19) == 0) {
1010 lmd->lmd_recovery_time_soft = max_t(int,
1011 simple_strtoul(s1 + 19, NULL, 10), time_min);
1012 clear++;
1013 } else if (strncmp(s1, "recovery_time_hard=", 19) == 0) {
1014 lmd->lmd_recovery_time_hard = max_t(int,
1015 simple_strtoul(s1 + 19, NULL, 10), time_min);
1016 clear++;
1017 } else if (strncmp(s1, "noir", 4) == 0) {
1018 lmd->lmd_flags |= LMD_FLG_NOIR; /* test purpose only. */
1019 clear++;
1020 } else if (strncmp(s1, "nosvc", 5) == 0) {
1021 lmd->lmd_flags |= LMD_FLG_NOSVC;
1022 clear++;
1023 } else if (strncmp(s1, "nomgs", 5) == 0) {
1024 lmd->lmd_flags |= LMD_FLG_NOMGS;
1025 clear++;
1026 } else if (strncmp(s1, "noscrub", 7) == 0) {
1027 lmd->lmd_flags |= LMD_FLG_NOSCRUB;
1028 clear++;
1029 } else if (strncmp(s1, PARAM_MGSNODE,
1030 sizeof(PARAM_MGSNODE) - 1) == 0) {
1031 s2 = s1 + sizeof(PARAM_MGSNODE) - 1;
1032 /* Assume the next mount opt is the first
1033 invalid nid we get to. */
1034 rc = lmd_parse_mgs(lmd, &s2);
1035 if (rc)
1036 goto invalid;
1037 clear++;
1038 } else if (strncmp(s1, "writeconf", 9) == 0) {
1039 lmd->lmd_flags |= LMD_FLG_WRITECONF;
1040 clear++;
1041 } else if (strncmp(s1, "update", 6) == 0) {
1042 lmd->lmd_flags |= LMD_FLG_UPDATE;
1043 clear++;
1044 } else if (strncmp(s1, "virgin", 6) == 0) {
1045 lmd->lmd_flags |= LMD_FLG_VIRGIN;
1046 clear++;
1047 } else if (strncmp(s1, "noprimnode", 10) == 0) {
1048 lmd->lmd_flags |= LMD_FLG_NO_PRIMNODE;
1049 clear++;
1050 } else if (strncmp(s1, "mgssec=", 7) == 0) {
1051 rc = lmd_parse_mgssec(lmd, s1 + 7);
1052 if (rc)
1053 goto invalid;
1054 clear++;
1055 /* ost exclusion list */
1056 } else if (strncmp(s1, "exclude=", 8) == 0) {
1057 rc = lmd_make_exclusion(lmd, s1 + 7);
1058 if (rc)
1059 goto invalid;
1060 clear++;
1061 } else if (strncmp(s1, "mgs", 3) == 0) {
1062 /* We are an MGS */
1063 lmd->lmd_flags |= LMD_FLG_MGS;
1064 clear++;
1065 } else if (strncmp(s1, "svname=", 7) == 0) {
1066 rc = lmd_parse_string(&lmd->lmd_profile, s1 + 7);
1067 if (rc)
1068 goto invalid;
1069 clear++;
1070 } else if (strncmp(s1, "param=", 6) == 0) {
1071 int length;
1072 char *tail = strchr(s1 + 6, ',');
1073 if (tail == NULL)
1074 length = strlen(s1);
1075 else
1076 length = tail - s1;
1077 length -= 6;
1078 strncat(lmd->lmd_params, s1 + 6, length);
1079 strcat(lmd->lmd_params, " ");
1080 clear++;
1081 } else if (strncmp(s1, "osd=", 4) == 0) {
1082 rc = lmd_parse_string(&lmd->lmd_osd_type, s1 + 4);
1083 if (rc)
1084 goto invalid;
1085 clear++;
1086 }
1087 /* Linux 2.4 doesn't pass the device, so we stuck it at the
1088 end of the options. */
1089 else if (strncmp(s1, "device=", 7) == 0) {
1090 devname = s1 + 7;
1091 /* terminate options right before device. device
1092 must be the last one. */
1093 *s1 = '\0';
1094 break;
1095 }
1096
1097 /* Find next opt */
1098 s2 = strchr(s1, ',');
1099 if (s2 == NULL) {
1100 if (clear)
1101 *s1 = '\0';
1102 break;
1103 }
1104 s2++;
1105 if (clear)
1106 memmove(s1, s2, strlen(s2) + 1);
1107 else
1108 s1 = s2;
1109 }
1110
1111 if (!devname) {
1112 LCONSOLE_ERROR_MSG(0x164, "Can't find the device name (need mount option 'device=...')\n");
1113 goto invalid;
1114 }
1115
1116 s1 = strstr(devname, ":/");
1117 if (s1) {
1118 ++s1;
1119 lmd->lmd_flags |= LMD_FLG_CLIENT;
1120 /* Remove leading /s from fsname */
1121 while (*++s1 == '/') ;
1122 /* Freed in lustre_free_lsi */
1123 lmd->lmd_profile = kzalloc(strlen(s1) + 8, GFP_NOFS);
1124 if (!lmd->lmd_profile)
1125 return -ENOMEM;
1126 sprintf(lmd->lmd_profile, "%s-client", s1);
1127 }
1128
1129 /* Freed in lustre_free_lsi */
1130 lmd->lmd_dev = kzalloc(strlen(devname) + 1, GFP_NOFS);
1131 if (!lmd->lmd_dev)
1132 return -ENOMEM;
1133 strcpy(lmd->lmd_dev, devname);
1134
1135 /* Save mount options */
1136 s1 = options + strlen(options) - 1;
1137 while (s1 >= options && (*s1 == ',' || *s1 == ' '))
1138 *s1-- = 0;
1139 if (*options != 0) {
1140 /* Freed in lustre_free_lsi */
1141 lmd->lmd_opts = kzalloc(strlen(options) + 1, GFP_NOFS);
1142 if (!lmd->lmd_opts)
1143 return -ENOMEM;
1144 strcpy(lmd->lmd_opts, options);
1145 }
1146
1147 lmd_print(lmd);
1148 lmd->lmd_magic = LMD_MAGIC;
1149
1150 return rc;
1151
1152 invalid:
1153 CERROR("Bad mount options %s\n", options);
1154 return -EINVAL;
1155 }
1156
1157 struct lustre_mount_data2 {
1158 void *lmd2_data;
1159 struct vfsmount *lmd2_mnt;
1160 };
1161
1162 /** This is the entry point for the mount call into Lustre.
1163 * This is called when a server or client is mounted,
1164 * and this is where we start setting things up.
1165 * @param data Mount options (e.g. -o flock,abort_recov)
1166 */
1167 int lustre_fill_super(struct super_block *sb, void *data, int silent)
1168 {
1169 struct lustre_mount_data *lmd;
1170 struct lustre_mount_data2 *lmd2 = data;
1171 struct lustre_sb_info *lsi;
1172 int rc;
1173
1174 CDEBUG(D_MOUNT|D_VFSTRACE, "VFS Op: sb %p\n", sb);
1175
1176 lsi = lustre_init_lsi(sb);
1177 if (!lsi)
1178 return -ENOMEM;
1179 lmd = lsi->lsi_lmd;
1180
1181 /*
1182 * Disable lockdep during mount, because mount locking patterns are
1183 * `special'.
1184 */
1185 lockdep_off();
1186
1187 /*
1188 * LU-639: the obd cleanup of last mount may not finish yet, wait here.
1189 */
1190 obd_zombie_barrier();
1191
1192 /* Figure out the lmd from the mount options */
1193 if (lmd_parse((char *)(lmd2->lmd2_data), lmd)) {
1194 lustre_put_lsi(sb);
1195 rc = -EINVAL;
1196 goto out;
1197 }
1198
1199 if (lmd_is_client(lmd)) {
1200 CDEBUG(D_MOUNT, "Mounting client %s\n", lmd->lmd_profile);
1201 if (client_fill_super == NULL)
1202 request_module("lustre");
1203 if (client_fill_super == NULL) {
1204 LCONSOLE_ERROR_MSG(0x165, "Nothing registered for client mount! Is the 'lustre' module loaded?\n");
1205 lustre_put_lsi(sb);
1206 rc = -ENODEV;
1207 } else {
1208 rc = lustre_start_mgc(sb);
1209 if (rc) {
1210 lustre_put_lsi(sb);
1211 goto out;
1212 }
1213 /* Connect and start */
1214 /* (should always be ll_fill_super) */
1215 rc = (*client_fill_super)(sb, lmd2->lmd2_mnt);
1216 /* c_f_s will call lustre_common_put_super on failure */
1217 }
1218 } else {
1219 CERROR("This is client-side-only module, cannot handle server mount.\n");
1220 rc = -EINVAL;
1221 }
1222
1223 /* If error happens in fill_super() call, @lsi will be killed there.
1224 * This is why we do not put it here. */
1225 goto out;
1226 out:
1227 if (rc) {
1228 CERROR("Unable to mount %s (%d)\n",
1229 s2lsi(sb) ? lmd->lmd_dev : "", rc);
1230 } else {
1231 CDEBUG(D_SUPER, "Mount %s complete\n",
1232 lmd->lmd_dev);
1233 }
1234 lockdep_on();
1235 return rc;
1236 }
1237
1238
1239 /* We can't call ll_fill_super by name because it lives in a module that
1240 must be loaded after this one. */
1241 void lustre_register_client_fill_super(int (*cfs)(struct super_block *sb,
1242 struct vfsmount *mnt))
1243 {
1244 client_fill_super = cfs;
1245 }
1246 EXPORT_SYMBOL(lustre_register_client_fill_super);
1247
1248 void lustre_register_kill_super_cb(void (*cfs)(struct super_block *sb))
1249 {
1250 kill_super_cb = cfs;
1251 }
1252 EXPORT_SYMBOL(lustre_register_kill_super_cb);
1253
1254 /***************** FS registration ******************/
1255 struct dentry *lustre_mount(struct file_system_type *fs_type, int flags,
1256 const char *devname, void *data)
1257 {
1258 struct lustre_mount_data2 lmd2 = {
1259 .lmd2_data = data,
1260 .lmd2_mnt = NULL
1261 };
1262
1263 return mount_nodev(fs_type, flags, &lmd2, lustre_fill_super);
1264 }
1265
1266 static void lustre_kill_super(struct super_block *sb)
1267 {
1268 struct lustre_sb_info *lsi = s2lsi(sb);
1269
1270 if (kill_super_cb && lsi && !IS_SERVER(lsi))
1271 (*kill_super_cb)(sb);
1272
1273 kill_anon_super(sb);
1274 }
1275
1276 /** Register the "lustre" fs type
1277 */
1278 struct file_system_type lustre_fs_type = {
1279 .owner = THIS_MODULE,
1280 .name = "lustre",
1281 .mount = lustre_mount,
1282 .kill_sb = lustre_kill_super,
1283 .fs_flags = FS_BINARY_MOUNTDATA | FS_REQUIRES_DEV |
1284 FS_HAS_FIEMAP | FS_RENAME_DOES_D_MOVE,
1285 };
1286 MODULE_ALIAS_FS("lustre");
1287
1288 int lustre_register_fs(void)
1289 {
1290 return register_filesystem(&lustre_fs_type);
1291 }
1292
1293 int lustre_unregister_fs(void)
1294 {
1295 return unregister_filesystem(&lustre_fs_type);
1296 }