mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-21 13:34:40 +03:00
57bb46c5e7
Filters are still applied before any device reading or the label scan, but any filter checks that want to read the device are skipped and the device is flagged. After bcache is populated, but before lvm looks for devices (i.e. before label scan), the filters are reapplied to the devices that were flagged above. The filters will then find the data they need in bcache.
113 lines
2.7 KiB
C
113 lines
2.7 KiB
C
/*
|
|
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2013 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "lib.h"
|
|
#include "filter.h"
|
|
#include "device.h"
|
|
|
|
static int _and_p(struct dev_filter *f, struct device *dev)
|
|
{
|
|
struct dev_filter **filters;
|
|
int ret;
|
|
|
|
for (filters = (struct dev_filter **) f->private; *filters; ++filters) {
|
|
ret = (*filters)->passes_filter(*filters, dev);
|
|
|
|
if (!ret)
|
|
return 0; /* No 'stack': a filter, not an error. */
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _and_p_with_dev_ext_info(struct dev_filter *f, struct device *dev)
|
|
{
|
|
int r;
|
|
|
|
dev_ext_enable(dev, external_device_info_source());
|
|
r = _and_p(f, dev);
|
|
dev_ext_disable(dev);
|
|
|
|
return r;
|
|
}
|
|
|
|
static void _composite_destroy(struct dev_filter *f)
|
|
{
|
|
struct dev_filter **filters;
|
|
|
|
if (f->use_count)
|
|
log_error(INTERNAL_ERROR "Destroying composite filter while in use %u times.", f->use_count);
|
|
|
|
for (filters = (struct dev_filter **) f->private; *filters; ++filters)
|
|
(*filters)->destroy(*filters);
|
|
|
|
dm_free(f->private);
|
|
dm_free(f);
|
|
}
|
|
|
|
static int _dump(struct dev_filter *f, int merge_existing)
|
|
{
|
|
struct dev_filter **filters;
|
|
|
|
for (filters = (struct dev_filter **) f->private; *filters; ++filters)
|
|
if ((*filters)->dump &&
|
|
!(*filters)->dump(*filters, merge_existing))
|
|
return_0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void _wipe(struct dev_filter *f)
|
|
{
|
|
struct dev_filter **filters;
|
|
|
|
for (filters = (struct dev_filter **) f->private; *filters; ++filters)
|
|
if ((*filters)->wipe)
|
|
(*filters)->wipe(*filters);
|
|
}
|
|
|
|
struct dev_filter *composite_filter_create(int n, int use_dev_ext_info, struct dev_filter **filters)
|
|
{
|
|
struct dev_filter **filters_copy, *cft;
|
|
|
|
if (!filters)
|
|
return_NULL;
|
|
|
|
if (!(filters_copy = dm_malloc(sizeof(*filters) * (n + 1)))) {
|
|
log_error("Composite filters allocation failed.");
|
|
return NULL;
|
|
}
|
|
|
|
memcpy(filters_copy, filters, sizeof(*filters) * n);
|
|
filters_copy[n] = NULL;
|
|
|
|
if (!(cft = dm_zalloc(sizeof(*cft)))) {
|
|
log_error("Composite filters allocation failed.");
|
|
dm_free(filters_copy);
|
|
return NULL;
|
|
}
|
|
|
|
cft->passes_filter = use_dev_ext_info ? _and_p_with_dev_ext_info : _and_p;
|
|
cft->destroy = _composite_destroy;
|
|
cft->dump = _dump;
|
|
cft->wipe = _wipe;
|
|
cft->use_count = 0;
|
|
cft->private = filters_copy;
|
|
|
|
log_debug_devs("Composite filter initialised.");
|
|
|
|
return cft;
|
|
}
|