#include #include #include #include #include "errno.h" #include "header.h" #include "misc.h" #include "oldheader.h" #include "package.h" #include "rpmerr.h" #include "rpmlead.h" #include "rpmlib.h" #include "signature.h" /* 0 = success */ /* !0 = error */ static int readOldHeader(int fd, Header * hdr, int * isSource); /* 0 = success */ /* 1 = bad magic */ /* 2 = error */ int pkgReadHeader(int fd, Header * hdr, int * isSource) { struct rpmlead lead; struct oldrpmlead * oldLead = (struct oldrpmlead *) &lead; int_8 arch; if (readLead(fd, &lead)) { return 2; } if (lead.magic[0] != RPMLEAD_MAGIC0 || lead.magic[1] != RPMLEAD_MAGIC1 || lead.magic[2] != RPMLEAD_MAGIC2 || lead.magic[3] != RPMLEAD_MAGIC3) { error(RPMERR_BADMAGIC, "file is not an RPM"); return 1; } *isSource = lead.type == RPMLEAD_SOURCE; if (*isSource) { message(MESS_DEBUG, "package is a source package major = %d\n", lead.major); if (lead.major == 1) { message(MESS_DEBUG, "archvie offset is 0x%x\n", oldLead->archiveOffset); oldLead->archiveOffset = ntohl(oldLead->archiveOffset); message(MESS_DEBUG, "archvie offset is %d\n", oldLead->archiveOffset); lseek(fd, oldLead->archiveOffset, SEEK_SET); /* we can't put togeher a header for old format source packages, there just isn't enough information there. We'll return NULL */ *hdr = NULL; } else { if (!readSignature(fd, lead.signature_type, NULL)) { return 2; } *hdr = readHeader(fd); if (! *hdr) return 2; } } else { if (lead.major == 1) { readOldHeader(fd, hdr, isSource); arch = getArchNum(); addEntry(*hdr, RPMTAG_ARCH, INT8_TYPE, &arch, 1); } else if (lead.major == 2) { if (!readSignature(fd, lead.signature_type, NULL)) { return 2; } *hdr = readHeader(fd); if (! *hdr) return 2; } else { error(RPMERR_NEWPACKAGE, "only packages with major numbers <= 2 are" " supported by this version of RPM"); return 2; } } return 0; } static int readOldHeader(int fd, Header * hdr, int * isSource) { struct oldrpmHeader oldheader; struct oldrpmHeaderSpec spec; Header dbentry; int_32 installTime = 0; char ** fileList; char ** fileMD5List; char ** fileLinktoList; int_32 * fileSizeList; int_32 * fileUIDList; int_32 * fileGIDList; int_32 * fileMtimesList; int_32 * fileFlagsList; int_16 * fileModesList; int_16 * fileRDevsList; char * fileStatesList; int i, j; lseek(fd, 0, SEEK_SET); if (oldhdrReadFromStream(fd, &oldheader)) { return 1; } if (oldhdrParseSpec(&oldheader, &spec)) { return 1; } dbentry = newHeader(); addEntry(dbentry, RPMTAG_NAME, STRING_TYPE, oldheader.name, 1); addEntry(dbentry, RPMTAG_VERSION, STRING_TYPE, oldheader.version, 1); addEntry(dbentry, RPMTAG_RELEASE, STRING_TYPE, oldheader.release, 1); addEntry(dbentry, RPMTAG_DESCRIPTION, STRING_TYPE, spec.description, 1); addEntry(dbentry, RPMTAG_BUILDTIME, INT32_TYPE, &spec.buildTime, 1); addEntry(dbentry, RPMTAG_BUILDHOST, STRING_TYPE, spec.buildHost, 1); addEntry(dbentry, RPMTAG_INSTALLTIME, INT32_TYPE, &installTime, 1); addEntry(dbentry, RPMTAG_DISTRIBUTION, STRING_TYPE, spec.distribution, 1); addEntry(dbentry, RPMTAG_VENDOR, STRING_TYPE, spec.vendor, 1); addEntry(dbentry, RPMTAG_SIZE, INT32_TYPE, &oldheader.size, 1); addEntry(dbentry, RPMTAG_COPYRIGHT, STRING_TYPE, spec.copyright, 1); if (oldheader.group) addEntry(dbentry, RPMTAG_GROUP, STRING_TYPE, oldheader.group, 1); else addEntry(dbentry, RPMTAG_GROUP, STRING_TYPE, "Unknown", 1); if (spec.prein) addEntry(dbentry, RPMTAG_PREIN, STRING_TYPE, spec.prein, 1); if (spec.preun) addEntry(dbentry, RPMTAG_PREUN, STRING_TYPE, spec.preun, 1); if (spec.postin) addEntry(dbentry, RPMTAG_POSTIN, STRING_TYPE, spec.postin, 1); if (spec.postun) addEntry(dbentry, RPMTAG_POSTUN, STRING_TYPE, spec.postun, 1); *hdr = dbentry; if (spec.fileCount) { /* some packages have no file lists */ fileList = malloc(sizeof(char *) * spec.fileCount); fileLinktoList = malloc(sizeof(char *) * spec.fileCount); fileMD5List = malloc(sizeof(char *) * spec.fileCount); fileSizeList = malloc(sizeof(int_32) * spec.fileCount); fileUIDList = malloc(sizeof(int_32) * spec.fileCount); fileGIDList = malloc(sizeof(int_32) * spec.fileCount); fileMtimesList = malloc(sizeof(int_32) * spec.fileCount); fileFlagsList = malloc(sizeof(int_32) * spec.fileCount); fileModesList = malloc(sizeof(int_16) * spec.fileCount); fileRDevsList = malloc(sizeof(int_16) * spec.fileCount); fileStatesList = malloc(sizeof(char) * spec.fileCount); /* old packages were reverse sorted, new ones are forward sorted */ j = spec.fileCount - 1; for (i = 0; i < spec.fileCount; i++, j--) { fileList[j] = spec.files[i].path; fileMD5List[j] = spec.files[i].md5; fileSizeList[j] = spec.files[i].size; fileUIDList[j] = spec.files[i].uid; fileGIDList[j] = spec.files[i].gid; fileMtimesList[j] = spec.files[i].mtime; fileModesList[j] = spec.files[i].mode; fileRDevsList[j] = spec.files[i].rdev; fileStatesList[j] = spec.files[i].state; if (spec.files[i].linkto) fileLinktoList[j] = spec.files[i].linkto; else fileLinktoList[j] = ""; fileFlagsList[j] = 0; if (spec.files[i].isdoc) fileFlagsList[j] |= RPMFILE_DOC; if (spec.files[i].isconf) fileFlagsList[j] |= RPMFILE_CONFIG; } addEntry(dbentry, RPMTAG_FILENAMES, STRING_ARRAY_TYPE, fileList, spec.fileCount); addEntry(dbentry, RPMTAG_FILELINKTOS, STRING_ARRAY_TYPE, fileLinktoList, spec.fileCount); addEntry(dbentry, RPMTAG_FILEMD5S, STRING_ARRAY_TYPE, fileMD5List, spec.fileCount); addEntry(dbentry, RPMTAG_FILESIZES, INT32_TYPE, fileSizeList, spec.fileCount); addEntry(dbentry, RPMTAG_FILEUIDS, INT32_TYPE, fileUIDList, spec.fileCount); addEntry(dbentry, RPMTAG_FILEGIDS, INT32_TYPE, fileGIDList, spec.fileCount); addEntry(dbentry, RPMTAG_FILEMTIMES, INT32_TYPE, fileMtimesList, spec.fileCount); addEntry(dbentry, RPMTAG_FILEFLAGS, INT32_TYPE, fileFlagsList, spec.fileCount); addEntry(dbentry, RPMTAG_FILEMODES, INT16_TYPE, fileModesList, spec.fileCount); addEntry(dbentry, RPMTAG_FILERDEVS, INT16_TYPE, fileRDevsList, spec.fileCount); addEntry(dbentry, RPMTAG_FILESTATES, INT8_TYPE, fileStatesList, spec.fileCount); free(fileList); free(fileLinktoList); free(fileMD5List); free(fileSizeList); free(fileUIDList); free(fileGIDList); free(fileMtimesList); free(fileFlagsList); free(fileModesList); free(fileRDevsList); free(fileStatesList); } oldhdrFree(&oldheader); return 0; }