diff --git a/src/Makefile.am b/src/Makefile.am index 3a968639a..30cda6f9c 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -24,6 +24,7 @@ bin_PROGRAMS = \ dpkg_SOURCES = \ archives.c archives.h \ + bencode.c \ cleanup.c \ configure.c \ depcon.c \ @@ -35,12 +36,14 @@ dpkg_SOURCES = \ help.c \ main.c main.h \ packages.c \ + pascal_str.c \ perpkgstate.c \ remove.c \ script.c \ select.c \ selinux.c \ trigproc.c \ + triggerer_info_passthrough.c \ unpack.c \ update.c \ verify.c \ diff --git a/src/archives.c b/src/archives.c index aa56e9e9c..bbea8447f 100644 --- a/src/archives.c +++ b/src/archives.c @@ -61,6 +61,7 @@ #include "main.h" #include "archives.h" #include "filters.h" +#include "triggerer_info_passthrough.h" static inline void fd_writeback_init(int fd) @@ -1478,6 +1479,7 @@ archivefiles(const char *const *argv) int i; jmp_buf ejbuf; enum modstatdb_rw msdbflags; + struct passed_through_package_info unpackedInfo = passed_through_package_info_init(); trigproc_install_hooks(); @@ -1580,7 +1582,7 @@ archivefiles(const char *const *argv) dpkg_selabel_load(); - process_archive(argp[i]); + process_archive(argp[i], &unpackedInfo); onerr_abort++; m_output(stdout, _("")); m_output(stderr, _("")); @@ -1588,6 +1590,8 @@ archivefiles(const char *const *argv) pop_error_context(ehflag_normaltidy); } + serialize_the_info_about_triggerers_into_an_env_variable(&unpackedInfo); + passed_through_package_info_free(&unpackedInfo); dpkg_selabel_close(); diff --git a/src/bencode.c b/src/bencode.c new file mode 100644 index 000000000..7f6b51a86 --- /dev/null +++ b/src/bencode.c @@ -0,0 +1,42 @@ +/* + * This is free and unencumbered software released into the public domain. + + * Anyone is free to copy, modify, publish, use, compile, sell, or distribute this software, either in source code form or as a compiled binary, for any purpose, commercial or non-commercial, and by any means. + + * In jurisdictions that recognize copyright laws, the author or authors of this software dedicate any and all copyright interest in the software to the public domain. We make this dedication for the benefit of the public at large and to the detriment of our heirs and successors. We intend this dedication to be an overt act of relinquishment in perpetuity of all present and future rights to this software under copyright law. + + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + + * For more information, please refer to + */ + +#include + +#include "bencode.h" + +size_t measure_benc_pascal_str(struct pascal_str *s){ + return snprintf(NULL, 0, "%zu:", s->size) + s->size; +} + +void serialize_benc_pascal_str(char **serPtr, struct pascal_str *s){ + *serPtr += sprintf(*serPtr, "%zu:%s", s->size, s->ptr); +} +void serialize_benc_tag(char **serPtr, char tag){ + *serPtr += sprintf(*serPtr, "1:%c", tag); +} + +void serialize_benc_tag_value_pair(char **serPtr, char tag, struct pascal_str *s){ + if(s->ptr){ + serialize_benc_tag(serPtr, tag); + serialize_benc_pascal_str(serPtr, s); + } +} + + +size_t measure_benc_tag_value_pair(struct pascal_str *s){ + if(s->ptr){ + return measure_benc_pascal_str(s) + OUR_BENC_KEY_TAG_SIZE; + } else { + return 0; + } +} diff --git a/src/bencode.h b/src/bencode.h new file mode 100644 index 000000000..20b541b44 --- /dev/null +++ b/src/bencode.h @@ -0,0 +1,26 @@ +#pragma once + +/* + * This is free and unencumbered software released into the public domain. + + * Anyone is free to copy, modify, publish, use, compile, sell, or distribute this software, either in source code form or as a compiled binary, for any purpose, commercial or non-commercial, and by any means. + + * In jurisdictions that recognize copyright laws, the author or authors of this software dedicate any and all copyright interest in the software to the public domain. We make this dedication for the benefit of the public at large and to the detriment of our heirs and successors. We intend this dedication to be an overt act of relinquishment in perpetuity of all present and future rights to this software under copyright law. + + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + + * For more information, please refer to + */ + +#include "pascal_str.h" + +enum { + OUR_BENC_KEY_TAG_SIZE = 3, + BENC_START_END = 2, // l<...>e +}; + +size_t measure_benc_pascal_str(struct pascal_str *s); +void serialize_benc_pascal_str(char **serPtr, struct pascal_str *s); +void serialize_benc_tag(char **serPtr, char tag); +void serialize_benc_tag_value_pair(char **serPtr, char tag, struct pascal_str *s); +size_t measure_benc_tag_value_pair(struct pascal_str *s); diff --git a/src/main.h b/src/main.h index e7fe820a7..17400e7c5 100644 --- a/src/main.h +++ b/src/main.h @@ -26,6 +26,7 @@ #include #include "force.h" +#include "triggerer_info_passthrough.h" /* These two are defined in . */ struct fsys_namenode_list; @@ -149,7 +150,7 @@ void ensure_package_clientdata(struct pkginfo *pkg); /* from archives.c */ int archivefiles(const char *const *argv); -void process_archive(const char *filename); +void process_archive(const char *filename, struct passed_through_package_info *unpackedInfo); bool wanttoinstall(struct pkginfo *pkg); /* from update.c */ diff --git a/src/packages.c b/src/packages.c index 9c4b75cc6..b457bc326 100644 --- a/src/packages.c +++ b/src/packages.c @@ -45,6 +45,7 @@ #include #include "main.h" +#include "triggerer_info_passthrough.h" static struct pkginfo *progress_bytrigproc; static struct pkg_queue queue = PKG_QUEUE_INIT; @@ -70,7 +71,7 @@ enqueue_package_mark_seen(struct pkginfo *pkg) } static void -enqueue_pending(void) +enqueue_pending(struct passed_through_package_info *packagePassedThroughInfo) { struct pkg_hash_iter *iter; struct pkginfo *pkg; @@ -108,13 +109,14 @@ enqueue_pending(void) default: internerr("unknown action '%d'", cipaction->arg_int); } + passed_through_package_info_append(packagePassedThroughInfo, pkg); enqueue_package(pkg); } pkg_hash_iter_free(iter); } static void -enqueue_specified(const char *const *argv) +enqueue_specified(const char *const *argv, struct passed_through_package_info *packagePassedThroughInfo) { const char *thisarg; @@ -122,6 +124,7 @@ enqueue_specified(const char *const *argv) struct pkginfo *pkg; pkg = dpkg_options_parse_pkgname(cipaction, thisarg); + passed_through_package_info_append(packagePassedThroughInfo, pkg); if (pkg->status == PKG_STAT_NOTINSTALLED && str_match_end(pkg->set->name, DEBEXT)) { badusage(_("you must specify packages by their own names, " @@ -137,6 +140,7 @@ enqueue_specified(const char *const *argv) int packages(const char *const *argv) { + struct passed_through_package_info unpackedInfo = passed_through_package_info_init(); trigproc_install_hooks(); modstatdb_open(f_noact ? msdbrw_readonly : @@ -151,14 +155,17 @@ packages(const char *const *argv) if (*argv) badusage(_("--%s --pending does not take any non-option arguments"),cipaction->olong); - enqueue_pending(); + enqueue_pending(&unpackedInfo); } else { if (!*argv) badusage(_("--%s needs at least one package name argument"), cipaction->olong); - enqueue_specified(argv); + enqueue_specified(argv, &unpackedInfo); } + serialize_the_info_about_triggerers_into_an_env_variable(&unpackedInfo); + passed_through_package_info_free(&unpackedInfo); + ensure_diversions(); process_queue(); diff --git a/src/pascal_str.c b/src/pascal_str.c new file mode 100644 index 000000000..1ef62434b --- /dev/null +++ b/src/pascal_str.c @@ -0,0 +1,42 @@ +/* + * This is free and unencumbered software released into the public domain. + + * Anyone is free to copy, modify, publish, use, compile, sell, or distribute this software, either in source code form or as a compiled binary, for any purpose, commercial or non-commercial, and by any means. + + * In jurisdictions that recognize copyright laws, the author or authors of this software dedicate any and all copyright interest in the software to the public domain. We make this dedication for the benefit of the public at large and to the detriment of our heirs and successors. We intend this dedication to be an overt act of relinquishment in perpetuity of all present and future rights to this software under copyright law. + + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + + * For more information, please refer to + */ + +#include +#include + +#include "pascal_str.h" + +struct pascal_str init_pascal_str(const char * s){ + struct pascal_str res; + if(s){ + res.size = strlen(s); + res.ptr = malloc(res.size + 1); + strcpy(res.ptr, s); + }else{ + res.size = 0; + res.ptr = NULL; + } + return res; +} + +struct pascal_str init_nonowning_pascal_str(const char * s){ + struct pascal_str res; + res.size = strlen(s); + res.ptr = s; + return res; +} + +void free_pascal_string(struct pascal_str *str){ + free(str->ptr); + str->ptr = NULL; + str->size = 0; +} diff --git a/src/pascal_str.h b/src/pascal_str.h new file mode 100644 index 000000000..1330eefd7 --- /dev/null +++ b/src/pascal_str.h @@ -0,0 +1,24 @@ +#pragma once + +/* + * This is free and unencumbered software released into the public domain. + + * Anyone is free to copy, modify, publish, use, compile, sell, or distribute this software, either in source code form or as a compiled binary, for any purpose, commercial or non-commercial, and by any means. + + * In jurisdictions that recognize copyright laws, the author or authors of this software dedicate any and all copyright interest in the software to the public domain. We make this dedication for the benefit of the public at large and to the detriment of our heirs and successors. We intend this dedication to be an overt act of relinquishment in perpetuity of all present and future rights to this software under copyright law. + + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + + * For more information, please refer to + */ + +struct pascal_str { + size_t size; + char * ptr; +}; + +struct pascal_str init_pascal_str(const char * s); + +struct pascal_str init_nonowning_pascal_str(const char * s); + +void free_pascal_string(struct pascal_str *str); diff --git a/src/triggerer_info_passthrough.c b/src/triggerer_info_passthrough.c new file mode 100644 index 000000000..c00897a24 --- /dev/null +++ b/src/triggerer_info_passthrough.c @@ -0,0 +1,114 @@ +#include + +#include "pascal_str.h" +#include "bencode.h" + +#include "triggerer_info_passthrough.h" +#include + + + +struct passed_through_package_info passed_through_package_info_init(void){ + struct passed_through_package_info unpackedInfo; + unpackedInfo.count = 0; + unpackedInfo.records = NULL; + return unpackedInfo; +} + + +void free_passed_through_package_info_record(struct passed_through_package_info_record *r){ + free_pascal_string(&r->name); + free_pascal_string(&r->arch); + free_pascal_string(&r->version); +} + +void passed_through_package_info_free(struct passed_through_package_info *unpackedInfo){ + for(unsigned short int i = 0; i< unpackedInfo->count; ++i){ + free_passed_through_package_info_record(&unpackedInfo->records[i]); + } + free(unpackedInfo->records); + unpackedInfo->records = NULL; + unpackedInfo->count = 0; +} + +void passed_through_package_info_emplace_internal(struct passed_through_package_info_record * rec, struct pkginfo *pkg, struct pascal_str (*pascal_str_ctor)(char *)){ + rec->name = pascal_str_ctor(pkg->set->name); + rec->version = pascal_str_ctor(pkg->available.version.version); + rec->arch = pascal_str_ctor(pkg->available.arch->name); + rec->origin = pascal_str_ctor(pkg->available.origin); +} + +void passed_through_package_info_emplace(struct passed_through_package_info_record * rec, struct pkginfo *pkg){ + passed_through_package_info_emplace_internal(rec, pkg, init_pascal_str); +} + +void passed_through_package_info_nonowning_emplace(struct passed_through_package_info_record * rec, struct pkginfo *pkg){ + passed_through_package_info_emplace_internal(rec, pkg, init_nonowning_pascal_str); +} + +void passed_through_package_info_append(struct passed_through_package_info *unpackedInfo, struct pkginfo *pkg){ + if(!unpackedInfo) + return; + + struct passed_through_package_info_record * rec; + unsigned int prevCountAndIndex = unpackedInfo->count; + unpackedInfo->records = realloc(unpackedInfo->records, sizeof(struct passed_through_package_info_record) * (++unpackedInfo->count)); + if(!unpackedInfo->records){ + ohshite("Unable to reallocate in %s", __func__); + } + rec = &(unpackedInfo->records[prevCountAndIndex]); + passed_through_package_info_emplace(rec, pkg); +} + +size_t precompute_the_bencode_serialized_size(struct passed_through_package_info *unpackedInfo){ + size_t totalSize = BENC_START_END; + unsigned int i; + for(i=0;icount;++i){ + if(unpackedInfo->records[i].name.size){ + totalSize += measure_benc_pascal_str(&unpackedInfo->records[i].name); + totalSize += BENC_START_END; + totalSize += measure_benc_tag_value_pair(&unpackedInfo->records[i].arch); + totalSize += measure_benc_tag_value_pair(&unpackedInfo->records[i].version); + totalSize += measure_benc_tag_value_pair(&unpackedInfo->records[i].origin); + } + } + totalSize ++; // null termination + return totalSize; +} + +void serialize_the_info_about_triggerers_into_an_env_variable(struct passed_through_package_info *unpackedInfo){ + unsigned int i; + size_t totalSize = precompute_the_bencode_serialized_size(unpackedInfo); + char buf4str[totalSize]; + char * serPtr; + + memset(buf4str, ' ', totalSize); + buf4str[totalSize - 1] = 0; + serPtr = &buf4str[0]; + *(serPtr++) = 'd'; + for(i=0;icount;++i){ + serialize_benc_pascal_str(&serPtr, &unpackedInfo->records[i].name); + *(serPtr++) = 'd'; + serialize_benc_tag_value_pair(&serPtr, 'A', &unpackedInfo->records[i].arch); + serialize_benc_tag_value_pair(&serPtr, 'V', &unpackedInfo->records[i].version); + serialize_benc_tag_value_pair(&serPtr, 'O', &unpackedInfo->records[i].origin); + *(serPtr++) = 'e'; + } + *(serPtr++) = 'e'; + if((unsigned long)((char*) serPtr - (char*) buf4str) > totalSize){ + ohshit("Buffer overflow %zu > %zu !", (char*) serPtr - (char*) buf4str, totalSize); + } + setenv("DPKG_TRIGGERER_PACKAGES_INFO", buf4str, 1); +} + +void serialize_the_info_about_pkg_into_an_env_variable(struct pkginfo *pkg){ + struct passed_through_package_info_record rec; // no need to free + //passed_through_package_info_append(unpackedInfo, pkg); + passed_through_package_info_emplace(&rec, pkg); + struct passed_through_package_info i; + i.records = &rec; + i.count = 1; + + passed_through_package_info_nonowning_emplace(&rec, pkg); + serialize_the_info_about_triggerers_into_an_env_variable(&i); +} diff --git a/src/triggerer_info_passthrough.h b/src/triggerer_info_passthrough.h new file mode 100644 index 000000000..afb075fc5 --- /dev/null +++ b/src/triggerer_info_passthrough.h @@ -0,0 +1,32 @@ +#pragma once +#include "pascal_str.h" + +#include "config.h" +#include + +struct passed_through_package_info_record { + struct pascal_str name; + struct pascal_str arch; + struct pascal_str version; + struct pascal_str origin; +}; + +struct passed_through_package_info{ + unsigned int count; + struct passed_through_package_info_record *records; +}; + +struct passed_through_package_info passed_through_package_info_init(void); + +void free_passed_through_package_info_record(struct passed_through_package_info_record *r); + +void passed_through_package_info_free(struct passed_through_package_info *unpackedInfo); + +void passed_through_package_info_emplace(struct passed_through_package_info_record * rec, struct pkginfo *pkg); + +void passed_through_package_info_append(struct passed_through_package_info *unpackedInfo, struct pkginfo *pkg); + +void serialize_the_info_about_triggerers_into_an_env_variable(struct passed_through_package_info *unpackedInfo); +void serialize_the_info_about_pkg_into_an_env_variable(struct pkginfo *pkg); + +size_t precompute_the_bencode_serialized_size(struct passed_through_package_info *unpackedInfo); diff --git a/src/unpack.c b/src/unpack.c index 76ef72586..c570f9f60 100644 --- a/src/unpack.c +++ b/src/unpack.c @@ -1092,7 +1092,8 @@ pkg_remove_backup_files(struct pkginfo *pkg, struct fsys_namenode_list *newfiles } } -void process_archive(const char *filename) { + +void process_archive(const char *filename, struct passed_through_package_info *unpackedInfo) { static const struct tar_operations tf = { .read = tarfileread, .extract_file = tarobject, @@ -1174,6 +1175,7 @@ void process_archive(const char *filename) { parsedb(cidir, parsedb_flags, &pkg); + passed_through_package_info_append(unpackedInfo, pkg); if (!pkg->archives) { pkg->archives = nfmalloc(sizeof(*pkg->archives)); pkg->archives->next = NULL;