storage_backend_scsi.c 19.8 KB
Newer Older
1 2 3
/*
 * storage_backend_scsi.c: storage backend for SCSI handling
 *
C
Cédric Bosdonnat 已提交
4
 * Copyright (C) 2007-2008, 2013-2014 Red Hat, Inc.
5 6 7 8 9 10 11 12 13 14 15 16 17
 * Copyright (C) 2007-2008 Daniel P. Berrange
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
18
 * License along with this library.  If not, see
O
Osier Yang 已提交
19
 * <http://www.gnu.org/licenses/>.
20 21 22 23 24 25 26 27 28 29 30
 *
 * Author: Daniel P. Berrange <berrange redhat com>
 */

#include <config.h>

#include <unistd.h>
#include <stdio.h>
#include <dirent.h>
#include <fcntl.h>

31
#include "virerror.h"
32
#include "storage_backend_scsi.h"
33
#include "viralloc.h"
34
#include "virlog.h"
E
Eric Blake 已提交
35
#include "virfile.h"
36
#include "vircommand.h"
37
#include "virstring.h"
38 39 40

#define VIR_FROM_THIS VIR_FROM_STORAGE

41 42
VIR_LOG_INIT("storage.storage_backend_scsi");

43 44 45 46 47
/* Function to check if the type file in the given sysfs_path is a
 * Direct-Access device (i.e. type 0).  Return -1 on failure, type of
 * the device otherwise.
 */
static int
48
getDeviceType(uint32_t host,
49 50 51 52 53 54 55 56 57 58 59 60
              uint32_t bus,
              uint32_t target,
              uint32_t lun,
              int *type)
{
    char *type_path = NULL;
    char typestr[3];
    char *gottype, *p;
    FILE *typefile;
    int retval = 0;

    if (virAsprintf(&type_path, "/sys/bus/scsi/devices/%u:%u:%u:%u/type",
61
                    host, bus, target, lun) < 0)
62 63 64 65
        goto out;

    typefile = fopen(type_path, "r");
    if (typefile == NULL) {
66
        virReportSystemError(errno,
67 68 69 70 71 72 73 74
                             _("Could not find typefile '%s'"),
                             type_path);
        /* there was no type file; that doesn't seem right */
        retval = -1;
        goto out;
    }

    gottype = fgets(typestr, 3, typefile);
75
    VIR_FORCE_FCLOSE(typefile);
76 77

    if (gottype == NULL) {
78
        virReportSystemError(errno,
79 80 81 82 83 84 85 86 87 88 89
                             _("Could not read typefile '%s'"),
                             type_path);
        /* we couldn't read the type file; have to give up */
        retval = -1;
        goto out;
    }

    /* we don't actually care about p, but if you pass NULL and the last
     * character is not \0, virStrToLong_i complains
     */
    if (virStrToLong_i(typestr, &p, 10, type) < 0) {
90 91 92
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Device type '%s' is not an integer"),
                       typestr);
93 94 95 96 97
        /* Hm, type wasn't an integer; seems strange */
        retval = -1;
        goto out;
    }

98
    VIR_DEBUG("Device type is %d", *type);
99

100
 out:
101 102 103 104
    VIR_FREE(type_path);
    return retval;
}

105 106 107 108
static char *
virStorageBackendSCSISerial(const char *dev)
{
    char *serial = NULL;
109
#ifdef WITH_UDEV
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129
    virCommandPtr cmd = virCommandNewArgList(
        "/lib/udev/scsi_id",
        "--replace-whitespace",
        "--whitelisted",
        "--device", dev,
        NULL
        );

    /* Run the program and capture its output */
    virCommandSetOutputBuffer(cmd, &serial);
    if (virCommandRun(cmd, NULL) < 0)
        goto cleanup;
#endif

    if (serial && STRNEQ(serial, "")) {
        char *nl = strchr(serial, '\n');
        if (nl)
            *nl = '\0';
    } else {
        VIR_FREE(serial);
130
        ignore_value(VIR_STRDUP(serial, dev));
131 132
    }

133
#ifdef WITH_UDEV
134
 cleanup:
135 136 137 138 139 140 141
    virCommandFree(cmd);
#endif

    return serial;
}


142
static int
143
virStorageBackendSCSINewLun(virStoragePoolObjPtr pool,
144
                            uint32_t host ATTRIBUTE_UNUSED,
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160
                            uint32_t bus,
                            uint32_t target,
                            uint32_t lun,
                            const char *dev)
{
    virStorageVolDefPtr vol;
    char *devpath = NULL;
    int retval = 0;

    if (VIR_ALLOC(vol) < 0) {
        retval = -1;
        goto out;
    }

    vol->type = VIR_STORAGE_VOL_BLOCK;

161 162 163 164 165 166
    /* 'host' is dynamically allocated by the kernel, first come,
     * first served, per HBA. As such it isn't suitable for use
     * in the volume name. We only need uniqueness per-pool, so
     * just leave 'host' out
     */
    if (virAsprintf(&(vol->name), "unit:%u:%u:%u", bus, target, lun) < 0) {
167 168 169 170 171 172 173 174 175
        retval = -1;
        goto free_vol;
    }

    if (virAsprintf(&devpath, "/dev/%s", dev) < 0) {
        retval = -1;
        goto free_vol;
    }

176
    VIR_DEBUG("Trying to create volume for '%s'", devpath);
177 178 179 180 181 182 183

    /* Now figure out the stable path
     *
     * XXX this method is O(N) because it scans the pool target
     * dir every time its run. Should figure out a more efficient
     * way of doing this...
     */
184
    if ((vol->target.path = virStorageBackendStablePath(pool,
185 186
                                                        devpath,
                                                        true)) == NULL) {
187 188 189 190
        retval = -1;
        goto free_vol;
    }

191
    if (STREQ(devpath, vol->target.path) &&
192 193 194
        !(STREQ(pool->def->target.path, "/dev") ||
          STREQ(pool->def->target.path, "/dev/"))) {

195
        VIR_DEBUG("No stable path found for '%s' in '%s'",
196 197 198 199 200 201
                  devpath, pool->def->target.path);

        retval = -1;
        goto free_vol;
    }

202
    if (virStorageBackendUpdateVolInfo(vol, true, true,
203
                                       VIR_STORAGE_VOL_OPEN_DEFAULT) < 0) {
204 205 206 207
        retval = -1;
        goto free_vol;
    }

208
    if (!(vol->key = virStorageBackendSCSISerial(vol->target.path))) {
209 210 211 212
        retval = -1;
        goto free_vol;
    }

213 214
    pool->def->capacity += vol->target.capacity;
    pool->def->allocation += vol->target.allocation;
215

216
    if (VIR_APPEND_ELEMENT(pool->volumes.objs, pool->volumes.count, vol) < 0) {
217 218 219 220 221 222
        retval = -1;
        goto free_vol;
    }

    goto out;

223
 free_vol:
224
    virStorageVolDefFree(vol);
225
 out:
226 227 228 229 230 231
    VIR_FREE(devpath);
    return retval;
}


static int
232
getNewStyleBlockDevice(const char *lun_path,
233 234 235 236 237 238 239
                       const char *block_name ATTRIBUTE_UNUSED,
                       char **block_device)
{
    char *block_path = NULL;
    DIR *block_dir = NULL;
    struct dirent *block_dirent = NULL;
    int retval = 0;
E
Eric Blake 已提交
240
    int direrr;
241

242
    if (virAsprintf(&block_path, "%s/block", lun_path) < 0)
243 244
        goto out;

245
    VIR_DEBUG("Looking for block device in '%s'", block_path);
246 247 248

    block_dir = opendir(block_path);
    if (block_dir == NULL) {
249
        virReportSystemError(errno,
250 251 252 253 254 255
                             _("Failed to opendir sysfs path '%s'"),
                             block_path);
        retval = -1;
        goto out;
    }

E
Eric Blake 已提交
256
    while ((direrr = virDirRead(block_dir, &block_dirent, block_path)) > 0) {
257 258 259 260 261

        if (STREQLEN(block_dirent->d_name, ".", 1)) {
            continue;
        }

262
        if (VIR_STRDUP(*block_device, block_dirent->d_name) < 0) {
263 264 265 266 267
            closedir(block_dir);
            retval = -1;
            goto out;
        }

268
        VIR_DEBUG("Block device is '%s'", *block_device);
269 270 271

        break;
    }
E
Eric Blake 已提交
272 273
    if (direrr < 0)
        retval = -1;
274 275 276

    closedir(block_dir);

277
 out:
278 279 280 281 282 283
    VIR_FREE(block_path);
    return retval;
}


static int
284
getOldStyleBlockDevice(const char *lun_path ATTRIBUTE_UNUSED,
285 286 287 288 289 290 291 292 293 294
                       const char *block_name,
                       char **block_device)
{
    char *blockp = NULL;
    int retval = 0;

    /* old-style; just parse out the sd */
    blockp = strrchr(block_name, ':');
    if (blockp == NULL) {
        /* Hm, wasn't what we were expecting; have to give up */
295 296 297
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Failed to parse block name %s"),
                       block_name);
298 299 300
        retval = -1;
    } else {
        blockp++;
301
        if (VIR_STRDUP(*block_device, blockp) < 0) {
302 303 304 305
            retval = -1;
            goto out;
        }

306
        VIR_DEBUG("Block device is '%s'", *block_device);
307 308
    }

309
 out:
310 311 312 313 314
    return retval;
}


static int
315
getBlockDevice(uint32_t host,
316 317 318 319 320 321 322 323 324
               uint32_t bus,
               uint32_t target,
               uint32_t lun,
               char **block_device)
{
    char *lun_path = NULL;
    DIR *lun_dir = NULL;
    struct dirent *lun_dirent = NULL;
    int retval = 0;
E
Eric Blake 已提交
325
    int direrr;
326 327

    if (virAsprintf(&lun_path, "/sys/bus/scsi/devices/%u:%u:%u:%u",
328
                    host, bus, target, lun) < 0)
329 330 331 332
        goto out;

    lun_dir = opendir(lun_path);
    if (lun_dir == NULL) {
333
        virReportSystemError(errno,
334 335 336 337 338 339
                             _("Failed to opendir sysfs path '%s'"),
                             lun_path);
        retval = -1;
        goto out;
    }

E
Eric Blake 已提交
340
    while ((direrr = virDirRead(lun_dir, &lun_dirent, lun_path)) > 0) {
341 342
        if (STREQLEN(lun_dirent->d_name, "block", 5)) {
            if (strlen(lun_dirent->d_name) == 5) {
343
                retval = getNewStyleBlockDevice(lun_path,
344 345 346
                                                lun_dirent->d_name,
                                                block_device);
            } else {
347
                retval = getOldStyleBlockDevice(lun_path,
348 349 350 351 352 353 354 355 356
                                                lun_dirent->d_name,
                                                block_device);
            }
            break;
        }
    }

    closedir(lun_dir);

357
 out:
358 359 360 361 362 363
    VIR_FREE(lun_path);
    return retval;
}


static int
364
processLU(virStoragePoolObjPtr pool,
365 366 367 368 369 370 371 372 373 374
          uint32_t host,
          uint32_t bus,
          uint32_t target,
          uint32_t lun)
{
    char *type_path = NULL;
    int retval = 0;
    int device_type;
    char *block_device = NULL;

375
    VIR_DEBUG("Processing LU %u:%u:%u:%u",
376 377
              host, bus, target, lun);

378
    if (getDeviceType(host, bus, target, lun, &device_type) < 0) {
379 380 381
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Failed to determine if %u:%u:%u:%u is a Direct-Access LUN"),
                       host, bus, target, lun);
382 383 384 385 386 387 388 389 390 391 392 393 394 395
        retval = -1;
        goto out;
    }

    /* We don't create volumes for devices other than disk and cdrom
     * devices, but finding a device that isn't one of those types
     * isn't an error, either. */
    if (!(device_type == VIR_STORAGE_DEVICE_TYPE_DISK ||
          device_type == VIR_STORAGE_DEVICE_TYPE_ROM))
    {
        retval = 0;
        goto out;
    }

396
    VIR_DEBUG("%u:%u:%u:%u is a Direct-Access LUN",
397 398
              host, bus, target, lun);

399
    if (getBlockDevice(host, bus, target, lun, &block_device) < 0) {
400 401 402
        goto out;
    }

403
    if (virStorageBackendSCSINewLun(pool,
404 405
                                    host, bus, target, lun,
                                    block_device) < 0) {
406
        VIR_DEBUG("Failed to create new storage volume for %u:%u:%u:%u",
407 408 409 410 411
                  host, bus, target, lun);
        retval = -1;
        goto out;
    }

412
    VIR_DEBUG("Created new storage volume for %u:%u:%u:%u successfully",
413 414 415 416
              host, bus, target, lun);

    VIR_FREE(type_path);

417
 out:
418
    VIR_FREE(block_device);
419 420 421 422 423
    return retval;
}


int
424
virStorageBackendSCSIFindLUs(virStoragePoolObjPtr pool,
425 426 427 428
                             uint32_t scanhost)
{
    int retval = 0;
    uint32_t bus, target, lun;
429
    const char *device_path = "/sys/bus/scsi/devices";
430 431 432
    DIR *devicedir = NULL;
    struct dirent *lun_dirent = NULL;
    char devicepattern[64];
433
    bool found = false;
434

435
    VIR_DEBUG("Discovering LUs on host %u", scanhost);
436

437
    virFileWaitForDevices();
438 439 440 441

    devicedir = opendir(device_path);

    if (devicedir == NULL) {
442
        virReportSystemError(errno,
443
                             _("Failed to opendir path '%s'"), device_path);
444
        return -1;
445 446 447 448
    }

    snprintf(devicepattern, sizeof(devicepattern), "%u:%%u:%%u:%%u\n", scanhost);

E
Eric Blake 已提交
449
    while ((retval = virDirRead(devicedir, &lun_dirent, device_path)) > 0) {
450 451 452 453 454
        if (sscanf(lun_dirent->d_name, devicepattern,
                   &bus, &target, &lun) != 3) {
            continue;
        }

455
        found = true;
456
        VIR_DEBUG("Found LU '%s'", lun_dirent->d_name);
457

458
        processLU(pool, scanhost, bus, target, lun);
459 460
    }

461 462 463
    if (!found)
        VIR_DEBUG("No LU found for pool %s", pool->def->name);

464 465 466 467 468
    closedir(devicedir);

    return retval;
}

469
static int
470
virStorageBackendSCSITriggerRescan(uint32_t host)
471 472 473 474 475
{
    int fd = -1;
    int retval = 0;
    char *path;

476
    VIR_DEBUG("Triggering rescan of host %d", host);
477

478 479
    if (virAsprintf(&path, "%s/host%u/scan",
                    LINUX_SYSFS_SCSI_HOST_PREFIX, host) < 0) {
480 481 482 483
        retval = -1;
        goto out;
    }

484
    VIR_DEBUG("Scan trigger path is '%s'", path);
485 486 487 488

    fd = open(path, O_WRONLY);

    if (fd < 0) {
489
        virReportSystemError(errno,
490 491 492 493 494 495 496 497 498
                             _("Could not open '%s' to trigger host scan"),
                             path);
        retval = -1;
        goto free_path;
    }

    if (safewrite(fd,
                  LINUX_SYSFS_SCSI_HOST_SCAN_STRING,
                  sizeof(LINUX_SYSFS_SCSI_HOST_SCAN_STRING)) < 0) {
499
        VIR_FORCE_CLOSE(fd);
500
        virReportSystemError(errno,
501 502 503 504 505
                             _("Write to '%s' to trigger host scan failed"),
                             path);
        retval = -1;
    }

506
    VIR_FORCE_CLOSE(fd);
507
 free_path:
508
    VIR_FREE(path);
509
 out:
510
    VIR_DEBUG("Rescan of host %d complete", host);
511 512 513
    return retval;
}

514 515 516 517
static char *
getAdapterName(virStoragePoolSourceAdapter adapter)
{
    char *name = NULL;
518
    char *parentaddr = NULL;
519

520
    if (adapter.type == VIR_STORAGE_POOL_SOURCE_ADAPTER_TYPE_SCSI_HOST) {
521
        if (adapter.data.scsi_host.has_parent) {
522
            virDevicePCIAddress addr = adapter.data.scsi_host.parentaddr;
523 524
            unsigned int unique_id = adapter.data.scsi_host.unique_id;

525 526 527 528 529
            if (!(name = virGetSCSIHostNameByParentaddr(addr.domain,
                                                        addr.bus,
                                                        addr.slot,
                                                        addr.function,
                                                        unique_id)))
530 531 532 533
                goto cleanup;
        } else {
            ignore_value(VIR_STRDUP(name, adapter.data.scsi_host.name));
        }
534 535 536 537 538 539 540 541 542
    } else if (adapter.type == VIR_STORAGE_POOL_SOURCE_ADAPTER_TYPE_FC_HOST) {
        if (!(name = virGetFCHostNameByWWN(NULL,
                                           adapter.data.fchost.wwnn,
                                           adapter.data.fchost.wwpn))) {
            virReportError(VIR_ERR_XML_ERROR,
                           _("Failed to find SCSI host with wwnn='%s', "
                             "wwpn='%s'"), adapter.data.fchost.wwnn,
                           adapter.data.fchost.wwpn);
        }
543 544
    }

545 546
 cleanup:
    VIR_FREE(parentaddr);
547 548 549
    return name;
}

550 551 552 553 554 555 556 557 558
static int
createVport(virStoragePoolSourceAdapter adapter)
{
    unsigned int parent_host;
    char *name = NULL;

    if (adapter.type != VIR_STORAGE_POOL_SOURCE_ADAPTER_TYPE_FC_HOST)
        return 0;

559 560 561 562 563 564 565 566 567 568 569 570 571 572
    /* If a parent was provided, then let's make sure it's vhost capable */
    if (adapter.data.fchost.parent) {
        if (virGetSCSIHostNumber(adapter.data.fchost.parent, &parent_host) < 0)
            return -1;

        if (!virIsCapableFCHost(NULL, parent_host)) {
            virReportError(VIR_ERR_XML_ERROR,
                           _("parent '%s' specified for vHBA "
                             "is not vport capable"),
                           adapter.data.fchost.parent);
            return -1;
        }
    }

573 574 575 576 577 578 579
    /* This filters either HBA or already created vHBA */
    if ((name = virGetFCHostNameByWWN(NULL, adapter.data.fchost.wwnn,
                                      adapter.data.fchost.wwpn))) {
        VIR_FREE(name);
        return 0;
    }

580 581 582 583 584 585 586
    if (!adapter.data.fchost.parent) {
        if (!(adapter.data.fchost.parent = virFindFCHostCapableVport(NULL))) {
            virReportError(VIR_ERR_XML_ERROR, "%s",
                           _("'parent' for vHBA not specified, and "
                             "cannot find one on this host"));
            return -1;
        }
587

588 589 590
        if (virGetSCSIHostNumber(adapter.data.fchost.parent, &parent_host) < 0)
            return -1;
    }
591

592 593
    if (virManageVport(parent_host, adapter.data.fchost.wwpn,
                       adapter.data.fchost.wwnn, VPORT_CREATE) < 0)
594 595 596 597 598 599 600 601 602 603
        return -1;

    virFileWaitForDevices();
    return 0;
}

static int
deleteVport(virStoragePoolSourceAdapter adapter)
{
    unsigned int parent_host;
O
Osier Yang 已提交
604 605
    char *name = NULL;
    int ret = -1;
606 607 608 609

    if (adapter.type != VIR_STORAGE_POOL_SOURCE_ADAPTER_TYPE_FC_HOST)
        return 0;

610 611 612 613 614 615 616 617
    /* It must be a HBA instead of a vHBA as long as "parent"
     * is NULL. "createVport" guaranteed "parent" for a vHBA
     * cannot be NULL, it's either specified in XML, or detected
     * automatically.
     */
    if (!adapter.data.fchost.parent)
        return 0;

O
Osier Yang 已提交
618 619
    if (!(name = virGetFCHostNameByWWN(NULL, adapter.data.fchost.wwnn,
                                       adapter.data.fchost.wwpn)))
620 621
        return -1;

622
    if (virGetSCSIHostNumber(adapter.data.fchost.parent, &parent_host) < 0)
O
Osier Yang 已提交
623
        goto cleanup;
624

625 626
    if (virManageVport(parent_host, adapter.data.fchost.wwpn,
                       adapter.data.fchost.wwnn, VPORT_DELETE) < 0)
O
Osier Yang 已提交
627
        goto cleanup;
628

O
Osier Yang 已提交
629
    ret = 0;
630
 cleanup:
O
Osier Yang 已提交
631 632
    VIR_FREE(name);
    return ret;
633 634 635
}


636 637 638 639 640
static int
virStorageBackendSCSICheckPool(virConnectPtr conn ATTRIBUTE_UNUSED,
                               virStoragePoolObjPtr pool,
                               bool *isActive)
{
641 642
    char *path = NULL;
    char *name = NULL;
643
    unsigned int host;
644
    int ret = -1;
645 646

    *isActive = false;
647

648 649 650 651 652 653 654 655 656 657 658 659 660
    if (!(name = getAdapterName(pool->def->source.adapter))) {
        /* It's normal for the pool with "fc_host" type source
         * adapter fails to get the adapter name, since the vHBA
         * the adapter based on might be not created yet.
         */
        if (pool->def->source.adapter.type ==
            VIR_STORAGE_POOL_SOURCE_ADAPTER_TYPE_FC_HOST) {
            virResetLastError();
            return 0;
        } else {
            return -1;
        }
    }
661

662
    if (virGetSCSIHostNumber(name, &host) < 0)
663 664
        goto cleanup;

665 666
    if (virAsprintf(&path, "%s/host%d",
                    LINUX_SYSFS_SCSI_HOST_PREFIX, host) < 0)
667
        goto cleanup;
668

669
    *isActive = virFileExists(path);
670

671
    ret = 0;
672
 cleanup:
673
    VIR_FREE(path);
674 675
    VIR_FREE(name);
    return ret;
676
}
677

678
static int
679
virStorageBackendSCSIRefreshPool(virConnectPtr conn ATTRIBUTE_UNUSED,
680 681
                                 virStoragePoolObjPtr pool)
{
682
    char *name = NULL;
683
    unsigned int host;
684
    int ret = -1;
685 686 687

    pool->def->allocation = pool->def->capacity = pool->def->available = 0;

688 689 690
    if (!(name = getAdapterName(pool->def->source.adapter)))
        return -1;

691
    if (virGetSCSIHostNumber(name, &host) < 0)
692 693
        goto out;

694
    VIR_DEBUG("Scanning host%u", host);
695

696
    if (virStorageBackendSCSITriggerRescan(host) < 0)
697 698
        goto out;

699
    virStorageBackendSCSIFindLUs(pool, host);
700

701
    ret = 0;
702
 out:
703
    VIR_FREE(name);
704
    return ret;
705 706
}

707 708 709 710 711 712 713 714 715 716 717 718 719 720 721
static int
virStorageBackendSCSIStartPool(virConnectPtr conn ATTRIBUTE_UNUSED,
                               virStoragePoolObjPtr pool)
{
    virStoragePoolSourceAdapter adapter = pool->def->source.adapter;
    return createVport(adapter);
}

static int
virStorageBackendSCSIStopPool(virConnectPtr conn ATTRIBUTE_UNUSED,
                              virStoragePoolObjPtr pool)
{
    virStoragePoolSourceAdapter adapter = pool->def->source.adapter;
    return deleteVport(adapter);
}
722 723 724 725

virStorageBackend virStorageBackendSCSI = {
    .type = VIR_STORAGE_POOL_SCSI,

726
    .checkPool = virStorageBackendSCSICheckPool,
727
    .refreshPool = virStorageBackendSCSIRefreshPool,
728 729
    .startPool = virStorageBackendSCSIStartPool,
    .stopPool = virStorageBackendSCSIStopPool,
730 731
    .uploadVol = virStorageBackendVolUploadLocal,
    .downloadVol = virStorageBackendVolDownloadLocal,
732
    .wipeVol = virStorageBackendVolWipeLocal,
733
};