summaryrefslogtreecommitdiff
path: root/lib/dpkg/dbmodify.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/dpkg/dbmodify.c')
-rw-r--r--lib/dpkg/dbmodify.c438
1 files changed, 438 insertions, 0 deletions
diff --git a/lib/dpkg/dbmodify.c b/lib/dpkg/dbmodify.c
new file mode 100644
index 0000000..61cfcb3
--- /dev/null
+++ b/lib/dpkg/dbmodify.c
@@ -0,0 +1,438 @@
+/*
+ * libdpkg - Debian packaging suite library routines
+ * dbmodify.c - routines for managing dpkg database updates
+ *
+ * Copyright © 1994,1995 Ian Jackson <ian@chiark.greenend.org.uk>
+ * Copyright © 2001 Wichert Akkerman <wichert@debian.org>
+ *
+ * This 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 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This 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 <compat.h>
+
+#include <sys/stat.h>
+#include <sys/types.h>
+#include <sys/wait.h>
+
+#include <assert.h>
+#include <errno.h>
+#include <limits.h>
+#include <ctype.h>
+#include <string.h>
+#include <time.h>
+#include <fcntl.h>
+#include <dirent.h>
+#include <unistd.h>
+#include <stdbool.h>
+#include <stdlib.h>
+#include <stdio.h>
+
+#include <dpkg/i18n.h>
+#include <dpkg/dpkg.h>
+#include <dpkg/dpkg-db.h>
+#include <dpkg/file.h>
+#include <dpkg/dir.h>
+#include <dpkg/triglib.h>
+
+static bool db_initialized;
+
+static enum modstatdb_rw cstatus=-1, cflags=0;
+static char *lockfile;
+static char *statusfile, *availablefile;
+static char *importanttmpfile=NULL;
+static FILE *importanttmp;
+static int nextupdate;
+static char *updatesdir;
+static int updateslength;
+static char *updatefnbuf, *updatefnrest;
+static struct varbuf uvb;
+
+static int ulist_select(const struct dirent *de) {
+ const char *p;
+ int l;
+ for (p= de->d_name, l=0; *p; p++, l++)
+ if (!cisdigit(*p)) return 0;
+ if (l > IMPORTANTMAXLEN)
+ ohshit(_("updates directory contains file `%.250s' whose name is too long "
+ "(length=%d, max=%d)"), de->d_name, l, IMPORTANTMAXLEN);
+ if (updateslength == -1) updateslength= l;
+ else if (l != updateslength)
+ ohshit(_("updates directory contains files with different length names "
+ "(both %d and %d)"), l, updateslength);
+ return 1;
+}
+
+static void cleanupdates(void) {
+ struct dirent **cdlist;
+ int cdn, i;
+
+ parsedb(statusfile, pdb_parse_status, NULL);
+
+ *updatefnrest = '\0';
+ updateslength= -1;
+ cdn= scandir(updatefnbuf, &cdlist, &ulist_select, alphasort);
+ if (cdn == -1) ohshite(_("cannot scan updates directory `%.255s'"),updatefnbuf);
+
+ if (cdn) {
+ for (i=0; i<cdn; i++) {
+ strcpy(updatefnrest, cdlist[i]->d_name);
+ parsedb(updatefnbuf, pdb_parse_update, NULL);
+ if (cstatus < msdbrw_write) free(cdlist[i]);
+ }
+
+ if (cstatus >= msdbrw_write) {
+ writedb(statusfile, wdb_must_sync);
+
+ for (i=0; i<cdn; i++) {
+ strcpy(updatefnrest, cdlist[i]->d_name);
+ if (unlink(updatefnbuf))
+ ohshite(_("failed to remove incorporated update file %.255s"),updatefnbuf);
+ free(cdlist[i]);
+ }
+
+ dir_sync_path(updatesdir);
+ }
+ }
+ free(cdlist);
+
+ nextupdate= 0;
+}
+
+static void createimptmp(void) {
+ int i;
+
+ onerr_abort++;
+
+ importanttmp= fopen(importanttmpfile,"w");
+ if (!importanttmp)
+ ohshite(_("unable to create `%.255s'"), importanttmpfile);
+ setcloexec(fileno(importanttmp),importanttmpfile);
+ for (i=0; i<512; i++) fputs("#padding\n",importanttmp);
+ if (ferror(importanttmp))
+ ohshite(_("unable to fill %.250s with padding"),importanttmpfile);
+ if (fflush(importanttmp))
+ ohshite(_("unable to flush %.250s after padding"), importanttmpfile);
+ if (fseek(importanttmp,0,SEEK_SET))
+ ohshite(_("unable to seek to start of %.250s after padding"),
+ importanttmpfile);
+
+ onerr_abort--;
+}
+
+static const struct fni {
+ const char *suffix;
+ char **store;
+} fnis[] = {
+ { LOCKFILE, &lockfile },
+ { STATUSFILE, &statusfile },
+ { AVAILFILE, &availablefile },
+ { UPDATESDIR, &updatesdir },
+ { UPDATESDIR IMPORTANTTMP, &importanttmpfile },
+ { NULL, NULL }
+};
+
+void
+modstatdb_init(void)
+{
+ const struct fni *fnip;
+
+ if (db_initialized)
+ return;
+
+ for (fnip = fnis; fnip->suffix; fnip++) {
+ free(*fnip->store);
+ *fnip->store = dpkg_db_get_path(fnip->suffix);
+ }
+
+ updatefnbuf = m_malloc(strlen(updatesdir) + IMPORTANTMAXLEN + 5);
+ strcpy(updatefnbuf, updatesdir);
+ updatefnrest = updatefnbuf + strlen(updatefnbuf);
+
+ db_initialized = true;
+}
+
+void
+modstatdb_done(void)
+{
+ const struct fni *fnip;
+
+ if (!db_initialized)
+ return;
+
+ for (fnip = fnis; fnip->suffix; fnip++) {
+ free(*fnip->store);
+ *fnip->store = NULL;
+ }
+ free(updatefnbuf);
+
+ db_initialized = false;
+}
+
+static int dblockfd = -1;
+
+bool
+modstatdb_is_locked(void)
+{
+ int lockfd;
+ bool locked;
+
+ if (dblockfd == -1) {
+ lockfd = open(lockfile, O_RDONLY);
+ if (lockfd == -1)
+ ohshite(_("unable to open lock file %s for testing"), lockfile);
+ } else {
+ lockfd = dblockfd;
+ }
+
+ locked = file_is_locked(lockfd, lockfile);
+
+ /* We only close the file if there was no lock open, otherwise we would
+ * release the existing lock on close. */
+ if (dblockfd == -1)
+ close(lockfd);
+
+ return locked;
+}
+
+bool
+modstatdb_can_lock(void)
+{
+ if (dblockfd >= 0)
+ return true;
+
+ dblockfd = open(lockfile, O_RDWR | O_CREAT | O_TRUNC, 0660);
+ if (dblockfd == -1) {
+ if (errno == EACCES || errno == EPERM)
+ return false;
+ else
+ ohshite(_("unable to open/create status database lockfile"));
+ }
+
+ return true;
+}
+
+void
+modstatdb_lock(void)
+{
+ if (!modstatdb_can_lock())
+ ohshit(_("you do not have permission to lock the dpkg status database"));
+
+ file_lock(&dblockfd, FILE_LOCK_NOWAIT, lockfile, _("dpkg status database"));
+}
+
+void
+modstatdb_unlock(void)
+{
+ /* Unlock. */
+ pop_cleanup(ehflag_normaltidy);
+
+ dblockfd = -1;
+}
+
+enum modstatdb_rw
+modstatdb_open(enum modstatdb_rw readwritereq)
+{
+ modstatdb_init();
+
+ cflags = readwritereq & msdbrw_available_mask;
+ readwritereq &= ~msdbrw_available_mask;
+
+ switch (readwritereq) {
+ case msdbrw_needsuperuser:
+ case msdbrw_needsuperuserlockonly:
+ if (getuid() || geteuid())
+ ohshit(_("requested operation requires superuser privilege"));
+ /* Fall through. */
+ case msdbrw_write: case msdbrw_writeifposs:
+ if (access(dpkg_db_get_dir(), W_OK)) {
+ if (errno != EACCES)
+ ohshite(_("unable to access dpkg status area"));
+ else if (readwritereq == msdbrw_write)
+ ohshit(_("operation requires read/write access to dpkg status area"));
+ cstatus= msdbrw_readonly;
+ } else {
+ modstatdb_lock();
+ cstatus= (readwritereq == msdbrw_needsuperuserlockonly ?
+ msdbrw_needsuperuserlockonly :
+ msdbrw_write);
+ }
+ break;
+ case msdbrw_readonly:
+ cstatus= msdbrw_readonly; break;
+ default:
+ internerr("unknown modstatdb_rw '%d'", readwritereq);
+ }
+
+ dpkg_arch_load_list();
+
+ if (cstatus != msdbrw_needsuperuserlockonly) {
+ cleanupdates();
+ if (cflags >= msdbrw_available_readonly)
+ parsedb(availablefile, pdb_parse_available, NULL);
+ }
+
+ if (cstatus >= msdbrw_write) {
+ createimptmp();
+ varbuf_init(&uvb, 10240);
+ }
+
+ trig_fixup_awaiters(cstatus);
+ trig_incorporate(cstatus);
+
+ return cstatus;
+}
+
+enum modstatdb_rw
+modstatdb_get_status(void)
+{
+ return cstatus;
+}
+
+void modstatdb_checkpoint(void) {
+ int i;
+
+ assert(cstatus >= msdbrw_write);
+ writedb(statusfile, wdb_must_sync);
+
+ for (i=0; i<nextupdate; i++) {
+ sprintf(updatefnrest, IMPORTANTFMT, i);
+ /* Have we made a real mess? */
+ assert(strlen(updatefnrest) <= IMPORTANTMAXLEN);
+ if (unlink(updatefnbuf))
+ ohshite(_("failed to remove my own update file %.255s"),updatefnbuf);
+ }
+
+ dir_sync_path(updatesdir);
+
+ nextupdate= 0;
+}
+
+void modstatdb_shutdown(void) {
+ if (cflags >= msdbrw_available_write)
+ writedb(availablefile, wdb_dump_available);
+
+ switch (cstatus) {
+ case msdbrw_write:
+ modstatdb_checkpoint();
+ /* Tidy up a bit, but don't worry too much about failure. */
+ fclose(importanttmp);
+ unlink(importanttmpfile);
+ varbuf_destroy(&uvb);
+ /* Fall through. */
+ case msdbrw_needsuperuserlockonly:
+ modstatdb_unlock();
+ default:
+ break;
+ }
+
+ modstatdb_done();
+}
+
+static void
+modstatdb_note_core(struct pkginfo *pkg)
+{
+ assert(cstatus >= msdbrw_write);
+
+ varbuf_reset(&uvb);
+ varbufrecord(&uvb, pkg, &pkg->installed);
+
+ if (fwrite(uvb.buf, 1, uvb.used, importanttmp) != uvb.used)
+ ohshite(_("unable to write updated status of `%.250s'"),
+ pkg_name(pkg, pnaw_nonambig));
+ if (fflush(importanttmp))
+ ohshite(_("unable to flush updated status of `%.250s'"),
+ pkg_name(pkg, pnaw_nonambig));
+ if (ftruncate(fileno(importanttmp), uvb.used))
+ ohshite(_("unable to truncate for updated status of `%.250s'"),
+ pkg_name(pkg, pnaw_nonambig));
+ if (fsync(fileno(importanttmp)))
+ ohshite(_("unable to fsync updated status of `%.250s'"),
+ pkg_name(pkg, pnaw_nonambig));
+ if (fclose(importanttmp))
+ ohshite(_("unable to close updated status of `%.250s'"),
+ pkg_name(pkg, pnaw_nonambig));
+ sprintf(updatefnrest, IMPORTANTFMT, nextupdate);
+ if (rename(importanttmpfile, updatefnbuf))
+ ohshite(_("unable to install updated status of `%.250s'"),
+ pkg_name(pkg, pnaw_nonambig));
+
+ dir_sync_path(updatesdir);
+
+ /* Have we made a real mess? */
+ assert(strlen(updatefnrest) <= IMPORTANTMAXLEN);
+
+ nextupdate++;
+
+ if (nextupdate > MAXUPDATES) {
+ modstatdb_checkpoint();
+ nextupdate = 0;
+ }
+
+ createimptmp();
+}
+
+/*
+ * Note: If anyone wants to set some triggers-pending, they must also
+ * set status appropriately, or we will undo it. That is, it is legal
+ * to call this when pkg->status and pkg->trigpend_head disagree and
+ * in that case pkg->status takes precedence and pkg->trigpend_head
+ * will be adjusted.
+ */
+void modstatdb_note(struct pkginfo *pkg) {
+ struct trigaw *ta;
+
+ onerr_abort++;
+
+ /* Clear pending triggers here so that only code that sets the status
+ * to interesting (for triggers) values has to care about triggers. */
+ if (pkg->status != stat_triggerspending &&
+ pkg->status != stat_triggersawaited)
+ pkg->trigpend_head = NULL;
+
+ if (pkg->status <= stat_configfiles) {
+ for (ta = pkg->trigaw.head; ta; ta = ta->sameaw.next)
+ ta->aw = NULL;
+ pkg->trigaw.head = pkg->trigaw.tail = NULL;
+ }
+
+ log_message("status %s %s %s", statusinfos[pkg->status].name,
+ pkg_name(pkg, pnaw_always),
+ versiondescribe(&pkg->installed.version, vdew_nonambig));
+ statusfd_send("status: %s: %s", pkg_name(pkg, pnaw_nonambig),
+ statusinfos[pkg->status].name);
+
+ if (cstatus >= msdbrw_write)
+ modstatdb_note_core(pkg);
+
+ if (!pkg->trigpend_head && pkg->othertrigaw_head) {
+ /* Automatically remove us from other packages' Triggers-Awaited.
+ * We do this last because we want to maximize our chances of
+ * successfully recording the status of the package we were
+ * pointed at by our caller, although there is some risk of
+ * leaving us in a slightly odd situation which is cleared up
+ * by the trigger handling logic in deppossi_ok_found. */
+ trig_clear_awaiters(pkg);
+ }
+
+ onerr_abort--;
+}
+
+void
+modstatdb_note_ifwrite(struct pkginfo *pkg)
+{
+ if (cstatus >= msdbrw_write)
+ modstatdb_note(pkg);
+}
+