/* * isys.c * * Copyright (C) 2007, 2008, 2009 Red Hat, Inc. All rights reserved. * * This program 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 program 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 . */ #include #include #include #include #define u32 __u32 #include #include /* Need to tell loop.h what the actual dev_t type is. */ #undef dev_t #if defined(__alpha) || (defined(__sparc__) && defined(__arch64__)) #define dev_t unsigned int #else #if defined(__x86_64__) #define dev_t unsigned long #else #define dev_t unsigned short #endif #endif #include #undef dev_t #define dev_t dev_t #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "iface.h" #include "isys.h" #include "imount.h" #include "ethtool.h" #include "lang.h" #include "eddsupport.h" #include "auditd.h" #include "imount.h" #include "log.h" #include "mem.h" #ifndef CDROMEJECT #define CDROMEJECT 0x5309 #endif static PyObject * doMount(PyObject * s, PyObject * args); static PyObject * doUMount(PyObject * s, PyObject * args); static PyObject * doSwapon(PyObject * s, PyObject * args); static PyObject * doSwapoff(PyObject * s, PyObject * args); static PyObject * doWipeRaidSuperblock(PyObject * s, PyObject * args); static PyObject * doGetRaidChunkSize(PyObject * s, PyObject * args); static PyObject * doDevSpaceFree(PyObject * s, PyObject * args); static PyObject * doExt2Dirty(PyObject * s, PyObject * args); static PyObject * doExt2HasJournal(PyObject * s, PyObject * args); static PyObject * doEjectCdrom(PyObject * s, PyObject * args); static PyObject * doisPseudoTTY(PyObject * s, PyObject * args); static PyObject * doisVioConsole(PyObject * s); static PyObject * doSync(PyObject * s, PyObject * args); static PyObject * doisIsoImage(PyObject * s, PyObject * args); static PyObject * printObject(PyObject * s, PyObject * args); static PyObject * py_bind_textdomain_codeset(PyObject * o, PyObject * args); static PyObject * doSegvHandler(PyObject *s, PyObject *args); static PyObject * doAuditDaemon(PyObject *s); static PyObject * doPrefixToNetmask(PyObject *s, PyObject *args); static PyObject * doGetBlkidData(PyObject * s, PyObject * args); static PyObject * doIsCapsLockEnabled(PyObject * s, PyObject * args); static PyObject * doGetLinkStatus(PyObject * s, PyObject * args); static PyObject * doGetAnacondaVersion(PyObject * s, PyObject * args); static PyObject * doInitLog(PyObject * s); static PyObject * doTotalMemory(PyObject * s); static PyMethodDef isysModuleMethods[] = { { "ejectcdrom", (PyCFunction) doEjectCdrom, METH_VARARGS, NULL }, { "e2dirty", (PyCFunction) doExt2Dirty, METH_VARARGS, NULL }, { "e2hasjournal", (PyCFunction) doExt2HasJournal, METH_VARARGS, NULL }, { "devSpaceFree", (PyCFunction) doDevSpaceFree, METH_VARARGS, NULL }, { "wiperaidsb", (PyCFunction) doWipeRaidSuperblock, METH_VARARGS, NULL }, { "getraidchunk", (PyCFunction) doGetRaidChunkSize, METH_VARARGS, NULL }, { "mount", (PyCFunction) doMount, METH_VARARGS, NULL }, { "umount", (PyCFunction) doUMount, METH_VARARGS, NULL }, { "swapon", (PyCFunction) doSwapon, METH_VARARGS, NULL }, { "swapoff", (PyCFunction) doSwapoff, METH_VARARGS, NULL }, { "isPseudoTTY", (PyCFunction) doisPseudoTTY, METH_VARARGS, NULL}, { "isVioConsole", (PyCFunction) doisVioConsole, METH_NOARGS, NULL}, { "sync", (PyCFunction) doSync, METH_VARARGS, NULL}, { "isisoimage", (PyCFunction) doisIsoImage, METH_VARARGS, NULL}, { "printObject", (PyCFunction) printObject, METH_VARARGS, NULL}, { "bind_textdomain_codeset", (PyCFunction) py_bind_textdomain_codeset, METH_VARARGS, NULL}, { "handleSegv", (PyCFunction) doSegvHandler, METH_VARARGS, NULL }, { "auditdaemon", (PyCFunction) doAuditDaemon, METH_NOARGS, NULL }, { "prefix2netmask", (PyCFunction) doPrefixToNetmask, METH_VARARGS, NULL }, { "getblkid", (PyCFunction) doGetBlkidData, METH_VARARGS, NULL }, { "isCapsLockEnabled", (PyCFunction) doIsCapsLockEnabled, METH_VARARGS, NULL }, { "getLinkStatus", (PyCFunction) doGetLinkStatus, METH_VARARGS, NULL }, { "getAnacondaVersion", (PyCFunction) doGetAnacondaVersion, METH_VARARGS, NULL }, { "initLog", (PyCFunction) doInitLog, METH_VARARGS, NULL }, { "total_memory", (PyCFunction) doTotalMemory, METH_NOARGS, NULL }, { NULL, NULL, 0, NULL } } ; static PyObject * doUMount(PyObject * s, PyObject * args) { char *err = NULL, *mntpoint = NULL; int rc; if (!PyArg_ParseTuple(args, "s", &mntpoint)) { return NULL; } Py_BEGIN_ALLOW_THREADS rc = doPwUmount(mntpoint, &err); Py_END_ALLOW_THREADS if (rc == IMOUNT_ERR_ERRNO) { PyErr_SetFromErrno(PyExc_SystemError); } else if (rc) { PyObject *tuple = PyTuple_New(2); PyTuple_SetItem(tuple, 0, PyInt_FromLong(rc)); if (err == NULL) { Py_INCREF(Py_None); PyTuple_SetItem(tuple, 1, Py_None); } else { PyTuple_SetItem(tuple, 1, PyString_FromString(err)); } PyErr_SetObject(PyExc_SystemError, tuple); } if (rc) { return NULL; } Py_INCREF(Py_None); return Py_None; } static PyObject * doMount(PyObject * s, PyObject * args) { char *err = NULL, *fs, *device, *mntpoint, *flags = NULL; int rc; if (!PyArg_ParseTuple(args, "sss|z", &fs, &device, &mntpoint, &flags)) return NULL; Py_BEGIN_ALLOW_THREADS rc = doPwMount(device, mntpoint, fs, flags, &err); Py_END_ALLOW_THREADS if (rc == IMOUNT_ERR_ERRNO) PyErr_SetFromErrno(PyExc_SystemError); else if (rc) { PyObject *tuple = PyTuple_New(2); PyTuple_SetItem(tuple, 0, PyInt_FromLong(rc)); if (err == NULL) { Py_INCREF(Py_None); PyTuple_SetItem(tuple, 1, Py_None); } else { PyTuple_SetItem(tuple, 1, PyString_FromString(err)); } PyErr_SetObject(PyExc_SystemError, tuple); } if (rc) return NULL; Py_INCREF(Py_None); return Py_None; } #define BOOT_SIGNATURE 0xaa55 /* boot signature */ #define BOOT_SIG_OFFSET 510 /* boot signature offset */ int swapoff(const char * path); int swapon(const char * path, int priorty); static PyObject * doSwapoff (PyObject * s, PyObject * args) { char * path; if (!PyArg_ParseTuple(args, "s", &path)) return NULL; if (swapoff (path)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } Py_INCREF(Py_None); return Py_None; } static PyObject * doSwapon (PyObject * s, PyObject * args) { char * path; if (!PyArg_ParseTuple(args, "s", &path)) return NULL; if (swapon (path, 0)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } Py_INCREF(Py_None); return Py_None; } void init_isys(void) { PyObject * m, * d; m = Py_InitModule("_isys", isysModuleMethods); d = PyModule_GetDict(m); PyDict_SetItemString(d, "MIN_RAM", PyInt_FromLong(MIN_RAM)); PyDict_SetItemString(d, "MIN_GUI_RAM", PyInt_FromLong(MIN_GUI_RAM)); PyDict_SetItemString(d, "GUI_INSTALL_EXTRA_RAM", PyInt_FromLong(GUI_INSTALL_EXTRA_RAM)); PyDict_SetItemString(d, "EARLY_SWAP_RAM", PyInt_FromLong(EARLY_SWAP_RAM)); } static PyObject * doPrefixToNetmask (PyObject * s, PyObject * args) { int prefix = 0; struct in_addr *mask = NULL; char dst[INET_ADDRSTRLEN+1]; if (!PyArg_ParseTuple(args, "i", &prefix)) return NULL; if ((mask = iface_prefix2netmask(prefix)) == NULL) return NULL; if (inet_ntop(AF_INET, mask, dst, INET_ADDRSTRLEN) == NULL) return NULL; return Py_BuildValue("s", dst); } static PyObject * doWipeRaidSuperblock(PyObject * s, PyObject * args) { int fd; unsigned long size; struct mdp_super_t * sb; if (!PyArg_ParseTuple(args, "i", &fd)) return NULL; if (ioctl(fd, BLKGETSIZE, &size)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } /* put the size in 1k blocks */ size >>= 1; if (lseek64(fd, ((off64_t) 512) * (off64_t) MD_NEW_SIZE_SECTORS(size), SEEK_SET) < 0) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } sb = malloc(sizeof(mdp_super_t)); sb = memset(sb, '\0', sizeof(mdp_super_t)); if (write(fd, sb, sizeof(sb)) != sizeof(sb)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } return Py_None; } static PyObject * doGetRaidChunkSize(PyObject * s, PyObject * args) { int fd; unsigned long size; mdp_super_t sb; if (!PyArg_ParseTuple(args, "i", &fd)) return NULL; if (ioctl(fd, BLKGETSIZE, &size)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } /* put the size in 1k blocks */ size >>= 1; if (lseek64(fd, ((off64_t) 512) * (off64_t) MD_NEW_SIZE_SECTORS(size), SEEK_SET) < 0) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } if (read(fd, &sb, sizeof(sb)) != sizeof(sb)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } if (sb.md_magic != MD_SB_MAGIC) { PyErr_SetString(PyExc_ValueError, "bad md magic on device"); return NULL; } return Py_BuildValue("i", sb.chunk_size / 1024); } static int get_bits(unsigned long long v) { int b = 0; if ( v & 0xffffffff00000000LLU ) { b += 32; v >>= 32; } if ( v & 0xffff0000LLU ) { b += 16; v >>= 16; } if ( v & 0xff00LLU ) { b += 8; v >>= 8; } if ( v & 0xf0LLU ) { b += 4; v >>= 4; } if ( v & 0xcLLU ) { b += 2; v >>= 2; } if ( v & 0x2LLU ) b++; return v ? b + 1 : b; } static PyObject * doDevSpaceFree(PyObject * s, PyObject * args) { char * path; struct statfs sb; unsigned long long size; if (!PyArg_ParseTuple(args, "s", &path)) return NULL; if (statfs(path, &sb)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } /* Calculate a saturated addition to prevent oveflow. */ if ( get_bits(sb.f_bfree) + get_bits(sb.f_bsize) <= 64 ) size = (unsigned long long)sb.f_bfree * sb.f_bsize; else size = ~0LLU; return PyLong_FromUnsignedLongLong(size>>20); } static PyObject * doExt2Dirty(PyObject * s, PyObject * args) { char * device; ext2_filsys fsys; int rc; int clean; if (!PyArg_ParseTuple(args, "s", &device)) return NULL; rc = ext2fs_open(device, EXT2_FLAG_FORCE, 0, 0, unix_io_manager, &fsys); if (rc) { Py_INCREF(Py_None); return Py_None; } clean = fsys->super->s_state & EXT2_VALID_FS; ext2fs_close(fsys); return Py_BuildValue("i", !clean); } static PyObject * doExt2HasJournal(PyObject * s, PyObject * args) { char * device; ext2_filsys fsys; int rc; int hasjournal; if (!PyArg_ParseTuple(args, "s", &device)) return NULL; rc = ext2fs_open(device, EXT2_FLAG_FORCE, 0, 0, unix_io_manager, &fsys); if (rc) { Py_INCREF(Py_None); return Py_None; } hasjournal = fsys->super->s_feature_compat & EXT3_FEATURE_COMPAT_HAS_JOURNAL; ext2fs_close(fsys); return Py_BuildValue("i", hasjournal); } static PyObject * doEjectCdrom(PyObject * s, PyObject * args) { int fd; if (!PyArg_ParseTuple(args, "i", &fd)) return NULL; /* Ask it to unlock the door and then eject the disc. There's really * nothing to do if unlocking doesn't work, so just eject without * worrying about it. -- pjones */ ioctl(fd, CDROM_LOCKDOOR, 0); if (ioctl(fd, CDROMEJECT, 1)) { PyErr_SetFromErrno(PyExc_SystemError); return NULL; } Py_INCREF(Py_None); return Py_None; } static PyObject * doisPseudoTTY(PyObject * s, PyObject * args) { int fd; struct stat sb; if (!PyArg_ParseTuple(args, "i", &fd)) return NULL; fstat(fd, &sb); /* XXX close enough for now */ return Py_BuildValue("i", ((major(sb.st_rdev) >= 136) && (major(sb.st_rdev) <= 143))); } static PyObject * doisVioConsole(PyObject * s) { return Py_BuildValue("i", isVioConsole()); } static PyObject * doSync(PyObject * s, PyObject * args) { int fd; if (!PyArg_ParseTuple(args, "", &fd)) return NULL; sync(); Py_INCREF(Py_None); return Py_None; } int fileIsIso(const char * file); static PyObject * doisIsoImage(PyObject * s, PyObject * args) { char * fn; int rc; if (!PyArg_ParseTuple(args, "s", &fn)) return NULL; rc = fileIsIso(fn); return Py_BuildValue("i", rc); } static PyObject * printObject (PyObject * o, PyObject * args) { PyObject * obj; char buf[256]; if (!PyArg_ParseTuple(args, "O", &obj)) return NULL; snprintf(buf, 256, "<%s object at %lx>", obj->ob_type->tp_name, (long) obj); return PyString_FromString(buf); } static PyObject * py_bind_textdomain_codeset(PyObject * o, PyObject * args) { char *domain, *codeset, *ret; if (!PyArg_ParseTuple(args, "ss", &domain, &codeset)) return NULL; ret = bind_textdomain_codeset(domain, codeset); if (ret) return PyString_FromString(ret); PyErr_SetFromErrno(PyExc_SystemError); return NULL; } static PyObject * doSegvHandler(PyObject *s, PyObject *args) { void *array[20]; size_t size; char **strings; size_t i; signal(SIGSEGV, SIG_DFL); /* back to default */ size = backtrace (array, 20); strings = backtrace_symbols (array, size); printf ("Anaconda received SIGSEGV!. Backtrace:\n"); for (i = 0; i < size; i++) printf ("%s\n", strings[i]); free (strings); exit(1); } static PyObject * doAuditDaemon(PyObject *s) { audit_daemonize(); Py_INCREF(Py_None); return Py_None; } static PyObject * doGetBlkidData(PyObject * s, PyObject * args) { char * dev, * key; blkid_cache cache; blkid_dev bdev = NULL; blkid_tag_iterate titer; const char * type, * data; if (!PyArg_ParseTuple(args, "ss", &dev, &key)) return NULL; blkid_get_cache(&cache, NULL); bdev = blkid_get_dev(cache, dev, BLKID_DEV_NORMAL); if (bdev == NULL) goto out; titer = blkid_tag_iterate_begin(bdev); while (blkid_tag_next(titer, &type, &data) >= 0) { if (!strcmp(type, key)) { blkid_tag_iterate_end(titer); return Py_BuildValue("s", data); } } blkid_tag_iterate_end(titer); out: Py_INCREF(Py_None); return Py_None; } static PyObject * doIsCapsLockEnabled(PyObject * s, PyObject * args) { Display *d = NULL; XkbStateRec state; if ((d = XOpenDisplay(NULL)) == NULL) { PyErr_SetString(PyExc_RuntimeError, "XOpenDisplay failed"); return NULL; } memset(&state, 0, sizeof(state)); XkbGetState(d, XkbUseCoreKbd, &state); if (XCloseDisplay(d)) { PyErr_SetString(PyExc_RuntimeError, "XCloseDisplay failed"); return NULL; } return PyBool_FromLong(state.locked_mods & LockMask); } static PyObject * doGetLinkStatus(PyObject * s, PyObject * args) { char *dev = NULL; if (!PyArg_ParseTuple(args, "s", &dev)) { return NULL; } if (get_link_status(dev) == 1) { return PyBool_FromLong(1); } return PyBool_FromLong(0); } static PyObject * doGetAnacondaVersion(PyObject * s, PyObject * args) { return Py_BuildValue("s", VERSION); } static PyObject * doInitLog(PyObject * s) { openLog(); Py_INCREF(Py_None); return Py_None; } static PyObject * doTotalMemory(PyObject * s) { unsigned long long tm = totalMemory(); return PyLong_FromUnsignedLongLong(tm); } /* vim:set shiftwidth=4 softtabstop=4: */