2009-10-15 17:20:41 +00:00
|
|
|
|
/* mountinfo.c - Track infos about mounts
|
|
|
|
|
* Copyright (C) 2009 Free Software Foundation, Inc.
|
|
|
|
|
*
|
|
|
|
|
* This file is part of GnuPG.
|
|
|
|
|
*
|
|
|
|
|
* GnuPG is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* GnuPG 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 General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
|
|
#include "g13.h"
|
|
|
|
|
#include "i18n.h"
|
|
|
|
|
#include "mountinfo.h"
|
|
|
|
|
|
|
|
|
|
#include "keyblob.h"
|
|
|
|
|
#include "utils.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The object to keep track of mount information. */
|
|
|
|
|
struct mounttable_s
|
|
|
|
|
{
|
|
|
|
|
int in_use; /* The slot is in use. */
|
|
|
|
|
char *container; /* Name of the container. */
|
|
|
|
|
char *mountpoint; /* Name of the mounttype. */
|
|
|
|
|
int conttype; /* Type of the container. */
|
|
|
|
|
unsigned int rid; /* Identifier of the runner task. */
|
2009-10-28 12:02:15 +00:00
|
|
|
|
struct {
|
|
|
|
|
unsigned int remove:1; /* True if the mountpoint shall be removed
|
|
|
|
|
on umount. */
|
|
|
|
|
} flags;
|
2009-10-15 17:20:41 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The allocated table of mounts and its size. */
|
|
|
|
|
static mtab_t mounttable;
|
|
|
|
|
size_t mounttable_size;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Add CONTAINER,MOUNTPOINT,CONTTYPE,RID to the mounttable. */
|
|
|
|
|
gpg_error_t
|
|
|
|
|
mountinfo_add_mount (const char *container, const char *mountpoint,
|
2009-10-28 12:02:15 +00:00
|
|
|
|
int conttype, unsigned int rid, int remove_flag)
|
2009-10-15 17:20:41 +00:00
|
|
|
|
{
|
|
|
|
|
size_t idx;
|
|
|
|
|
mtab_t m;
|
|
|
|
|
|
|
|
|
|
for (idx=0; idx < mounttable_size; idx++)
|
|
|
|
|
if (!mounttable[idx].in_use)
|
|
|
|
|
break;
|
|
|
|
|
if (!(idx < mounttable_size))
|
|
|
|
|
{
|
|
|
|
|
size_t nslots = mounttable_size;
|
|
|
|
|
|
|
|
|
|
mounttable_size += 10;
|
|
|
|
|
m = xtrycalloc (mounttable_size, sizeof *mounttable);
|
|
|
|
|
if (!m)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
if (mounttable)
|
|
|
|
|
{
|
|
|
|
|
for (idx=0; idx < nslots; idx++)
|
|
|
|
|
m[idx] = mounttable[idx];
|
|
|
|
|
xfree (mounttable);
|
|
|
|
|
}
|
|
|
|
|
mounttable = m;
|
|
|
|
|
m = mounttable + nslots;
|
|
|
|
|
assert (!m->in_use);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
m = mounttable + idx;
|
|
|
|
|
|
|
|
|
|
m->container = xtrystrdup (container);
|
|
|
|
|
if (!m->container)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
m->mountpoint = xtrystrdup (mountpoint);
|
|
|
|
|
if (!m->mountpoint)
|
|
|
|
|
{
|
|
|
|
|
xfree (m->container);
|
|
|
|
|
m->container = NULL;
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
}
|
|
|
|
|
m->conttype = conttype;
|
|
|
|
|
m->rid = rid;
|
2009-10-28 12:02:15 +00:00
|
|
|
|
m->flags.remove = !!remove_flag;
|
2009-10-15 17:20:41 +00:00
|
|
|
|
m->in_use = 1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Remove a mount info. Either the CONTAINER, the MOUNTPOINT or the
|
|
|
|
|
RID must be given. The first argument given is used. */
|
|
|
|
|
gpg_error_t
|
|
|
|
|
mountinfo_del_mount (const char *container, const char *mountpoint,
|
|
|
|
|
unsigned int rid)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
size_t idx;
|
|
|
|
|
mtab_t m;
|
|
|
|
|
|
|
|
|
|
/* If a container or mountpint is givem search the RID via the
|
|
|
|
|
standard find fucntion. */
|
|
|
|
|
if (container || mountpoint)
|
|
|
|
|
{
|
|
|
|
|
err = mountinfo_find_mount (container, mountpoint, &rid);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Find via RID and delete. */
|
|
|
|
|
for (idx=0, m = mounttable; idx < mounttable_size; idx++, m++)
|
|
|
|
|
if (m->in_use && m->rid == rid)
|
|
|
|
|
{
|
2009-10-28 12:02:15 +00:00
|
|
|
|
if (m->flags.remove && m->mountpoint)
|
|
|
|
|
{
|
|
|
|
|
/* FIXME: This does not always work because the umount may
|
|
|
|
|
not have completed yet. We should add the mountpoints
|
2011-02-04 12:57:53 +01:00
|
|
|
|
to an idle queue and retry a remove. */
|
2009-10-28 12:02:15 +00:00
|
|
|
|
if (rmdir (m->mountpoint))
|
|
|
|
|
log_error ("error removing mount point `%s': %s\n",
|
2011-02-04 12:57:53 +01:00
|
|
|
|
m->mountpoint,
|
2009-10-28 12:02:15 +00:00
|
|
|
|
gpg_strerror (gpg_error_from_syserror ()));
|
|
|
|
|
}
|
2009-10-15 17:20:41 +00:00
|
|
|
|
m->in_use = 0;
|
|
|
|
|
xfree (m->container);
|
|
|
|
|
m->container = NULL;
|
|
|
|
|
xfree (m->mountpoint);
|
|
|
|
|
m->mountpoint = NULL;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Find a mount and return its rid at R_RID. If CONTAINER is given,
|
|
|
|
|
the search is done by the container name, if it is not given the
|
|
|
|
|
search is done by MOUNTPOINT. */
|
|
|
|
|
gpg_error_t
|
|
|
|
|
mountinfo_find_mount (const char *container, const char *mountpoint,
|
|
|
|
|
unsigned int *r_rid)
|
|
|
|
|
{
|
|
|
|
|
size_t idx;
|
|
|
|
|
mtab_t m;
|
|
|
|
|
|
|
|
|
|
if (container)
|
|
|
|
|
{
|
|
|
|
|
for (idx=0, m = mounttable; idx < mounttable_size; idx++, m++)
|
|
|
|
|
if (m->in_use && !strcmp (m->container, container))
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else if (mountpoint)
|
|
|
|
|
{
|
|
|
|
|
for (idx=0, m = mounttable; idx < mounttable_size; idx++, m++)
|
|
|
|
|
if (m->in_use && !strcmp (m->mountpoint, mountpoint))
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
idx = mounttable_size;
|
|
|
|
|
if (!(idx < mounttable_size))
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
|
|
|
|
|
*r_rid = m->rid;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Dump all info to the log stream. */
|
|
|
|
|
void
|
|
|
|
|
mountinfo_dump_all (void)
|
|
|
|
|
{
|
|
|
|
|
size_t idx;
|
|
|
|
|
mtab_t m;
|
|
|
|
|
|
|
|
|
|
for (idx=0, m = mounttable; idx < mounttable_size; idx++, m++)
|
|
|
|
|
if (m->in_use)
|
2011-02-04 12:57:53 +01:00
|
|
|
|
log_info ("mtab[%d] %s on %s type %d rid %u%s\n",
|
2009-10-28 12:02:15 +00:00
|
|
|
|
idx, m->container, m->mountpoint, m->conttype, m->rid,
|
|
|
|
|
m->flags.remove?" [remove]":"");
|
2009-10-15 17:20:41 +00:00
|
|
|
|
}
|