[dm-devel] [PATCH] kpartx: Improve finding loopback device by file
Julian Andres Klode
julian.klode at canonical.com
Mon Feb 5 10:45:00 UTC 2018
On Mon, Feb 05, 2018 at 11:32:13AM +0100, Martin Wilck wrote:
> On Mon, 2018-02-05 at 09:58 +0100, Julian Andres Klode wrote:
> > Commit 9bdfa3eb8e24b668e6c2bb882cddb0ccfe23ed5b changed kpartx
> > to lookup files by absolute path, using realpath(). This introduced
> > a regression when part of the filename where symbolic links. While
> > the kernel stores the absolute path to the backing file, it does not
> > resolve symbolic links, and hence kpartx would fail to find the
> > loopback
> > device because it resolves symbolic links when deleting.
> >
> > This introduces two workarounds in find_loop_by_file()
> >
> > (1) We match against the specified file name as is as well
> > (2) We canonicalize the loopinfo.lo_name and match the canonicalized
> > filename
> > against that.
> >
> > Bug-Ubuntu: https://bugs.launchpad.net/ubuntu/+source/multipath-tools
> > /+bug/1747044
> > Signed-off-by: Julian Andres Klode <julian.klode at canonical.com>
> > ---
> > kpartx/kpartx.c | 8 +-------
> > kpartx/lopart.c | 12 +++++++++++-
> > 2 files changed, 12 insertions(+), 8 deletions(-)
> >
> > diff --git a/kpartx/kpartx.c b/kpartx/kpartx.c
> > index 442b6bd9..c1af1c5e 100644
> > --- a/kpartx/kpartx.c
> > +++ b/kpartx/kpartx.c
> > @@ -327,13 +327,7 @@ main(int argc, char **argv){
> >
> > if (S_ISREG (buf.st_mode)) {
> > /* already looped file ? */
> > - char rpath[PATH_MAX];
> > - if (realpath(device, rpath) == NULL) {
> > - fprintf(stderr, "Error: %s: %s\n", device,
> > - strerror(errno));
> > - exit (1);
> > - }
> > - loopdev = find_loop_by_file(rpath);
> > + loopdev = find_loop_by_file(device);
> >
> > if (!loopdev && what == DELETE)
> > exit (0);
> > diff --git a/kpartx/lopart.c b/kpartx/lopart.c
> > index 02b29e8c..26b2678c 100644
> > --- a/kpartx/lopart.c
> > +++ b/kpartx/lopart.c
> > @@ -69,6 +69,13 @@ char *find_loop_by_file(const char *filename)
> > struct loop_info loopinfo;
> > const char VIRT_BLOCK[] = "/sys/devices/virtual/block";
> > char path[PATH_MAX];
> > + char rfilename[PATH_MAX];
> > + char rloopfilename[PATH_MAX];
> > + if (realpath(filename, rfilename) == NULL) {
> > + fprintf(stderr, "Error: %s: %s\n", filename,
> > + strerror(errno));
> > + exit (1);
> > + }
> >
> > dir = opendir(VIRT_BLOCK);
> > if (!dir)
> > @@ -117,7 +124,10 @@ char *find_loop_by_file(const char *filename)
> >
> > close (fd);
> >
> > - if (0 == strcmp(filename, loopinfo.lo_name)) {
> > + if (0 == strcmp(filename, loopinfo.lo_name) ||
> > + 0 == strcmp(rfilename, loopinfo.lo_name) ||
> > + (realpath(loopinfo.lo_name, rloopfilename) &&
> > + 0 == strcmp(rfilename, rloopfilename))) {
> > found = realpath(path, NULL);
> > break;
> > }
>
> That "if x matches y or x matches y' or x' matches y" feels like
> guesswork. Can't we simply call realpath() on both loopinfo.lo_name and
> filename, and compare the two?
Probably yes. That said there might be some corner cases:
(1) What happens if the backing file of a loopback is deleted - realpath
can fail with ENOENT.
(2) A path could be longer than PATH_MAX and it would fail with ENAMETOOLONG
- this is a "problem" with the initial realpath as well, but less so, because
the user can control that.
I guess the 0 == strcmp(rfilename, loopinfo.lo_name) is not likely to happen, but
in the end, I did not want to think that much about it and go with something that
definitely works as best as possible.
--
debian developer - deb.li/jak | jak-linux.org - free software dev
ubuntu core developer i speak de, en
More information about the dm-devel
mailing list