split.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. /*
  2. * dpkg-split - splitting and joining of multipart *.deb archives
  3. * split.c - splitting archives
  4. *
  5. * Copyright © 1995 Ian Jackson <ian@chiark.greenend.org.uk>
  6. * Copyright © 2010 Guillem Jover <guillem@debian.org>
  7. *
  8. * This is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation; either version 2 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. */
  21. #include <config.h>
  22. #include <compat.h>
  23. #include <sys/types.h>
  24. #include <sys/stat.h>
  25. #include <sys/wait.h>
  26. #include <limits.h>
  27. #include <errno.h>
  28. #include <fcntl.h>
  29. #include <libgen.h>
  30. #include <string.h>
  31. #include <ctype.h>
  32. #include <time.h>
  33. #include <unistd.h>
  34. #include <stdlib.h>
  35. #include <stdio.h>
  36. #include <dpkg/i18n.h>
  37. #include <dpkg/dpkg.h>
  38. #include <dpkg/dpkg-db.h>
  39. #include <dpkg/subproc.h>
  40. #include <dpkg/buffer.h>
  41. #include <dpkg/ar.h>
  42. #include <dpkg/myopt.h>
  43. #include "dpkg-split.h"
  44. static char *
  45. deb_field(const char *filename, const char *field)
  46. {
  47. pid_t pid;
  48. int p[2];
  49. struct varbuf buf = VARBUF_INIT;
  50. char *end;
  51. m_pipe(p);
  52. pid = subproc_fork();
  53. if (pid == 0) {
  54. /* Child writes to pipe. */
  55. m_dup2(p[1], 1);
  56. close(p[0]);
  57. close(p[1]);
  58. execlp(BACKEND, BACKEND, "--field", filename, field, NULL);
  59. ohshite(_("failed to exec dpkg-deb to extract field value"));
  60. }
  61. close(p[1]);
  62. /* Parant reads from pipe. */
  63. varbufreset(&buf);
  64. fd_vbuf_copy(p[0], &buf, -1, _("dpkg-deb field extraction"));
  65. varbufaddc(&buf, '\0');
  66. close(p[0]);
  67. subproc_wait_check(pid, _("dpkg-deb field extraction"), PROCPIPE);
  68. /* Trim down trailing junk. */
  69. for (end = buf.buf + strlen(buf.buf) - 1; end - buf.buf >= 1; end--)
  70. if (isspace(*end))
  71. *end = '\0';
  72. else
  73. break;
  74. return varbuf_detach(&buf);
  75. }
  76. /* Cleanup filename for use in crippled msdos systems. */
  77. static char *
  78. clean_msdos_filename(char *filename)
  79. {
  80. char *d, *s;
  81. for (s = d = filename; *s; d++, s++) {
  82. if (*s == '+')
  83. *d = 'x';
  84. else if (isupper(*s))
  85. *d = tolower(*s);
  86. else if (islower(*s) || isdigit(*s))
  87. *d = *s;
  88. else
  89. s++;
  90. }
  91. return filename;
  92. }
  93. static int
  94. mksplit(const char *file_src, const char *prefix, size_t partsize,
  95. size_t maxpartsize, bool msdos)
  96. {
  97. int fd_src;
  98. struct stat st;
  99. char hash[MD5HASHLEN + 1];
  100. char *package, *version;
  101. int nparts, curpart;
  102. off_t startat;
  103. char *prefixdir = NULL, *msdos_prefix = NULL;
  104. struct varbuf file_dst = VARBUF_INIT;
  105. struct varbuf partmagic = VARBUF_INIT;
  106. struct varbuf partname = VARBUF_INIT;
  107. char *partdata;
  108. fd_src = open(file_src, O_RDONLY);
  109. if (fd_src < 0)
  110. ohshite(_("unable to open source file `%.250s'"), file_src);
  111. if (fstat(fd_src, &st))
  112. ohshite(_("unable to fstat source file"));
  113. if (!S_ISREG(st.st_mode))
  114. ohshit(_("source file `%.250s' not a plain file"), file_src);
  115. fd_md5(fd_src, hash, -1, "md5hash");
  116. lseek(fd_src, 0, SEEK_SET);
  117. /* FIXME: Use libdpkg-deb. */
  118. package = deb_field(file_src, "Package");
  119. version = deb_field(file_src, "Version");
  120. nparts = (st.st_size + partsize - 1) / partsize;
  121. setvbuf(stdout, NULL, _IONBF, 0);
  122. printf("Splitting package %s into %d parts: ", package, nparts);
  123. if (msdos) {
  124. char *t;
  125. t = m_strdup(prefix);
  126. prefixdir = m_strdup(dirname(t));
  127. free(t);
  128. t = m_strdup(prefix);
  129. msdos_prefix = m_strdup(basename(t));
  130. free(t);
  131. prefix = clean_msdos_filename(msdos_prefix);
  132. }
  133. partdata = m_malloc(partsize);
  134. curpart = 1;
  135. for (startat = 0; startat < st.st_size; startat += partsize) {
  136. int fd_dst;
  137. ssize_t partrealsize;
  138. varbufreset(&file_dst);
  139. /* Generate output filename. */
  140. if (msdos) {
  141. struct varbuf refname = VARBUF_INIT;
  142. int prefix_max;
  143. varbufprintf(&refname, "%dof%d", curpart, nparts);
  144. prefix_max = max(8 - strlen(refname.buf), 0);
  145. varbufprintf(&file_dst, "%s/%.*s%.8s.deb",
  146. prefixdir, prefix_max, prefix,
  147. refname.buf);
  148. varbuf_destroy(&refname);
  149. } else {
  150. varbufprintf(&file_dst, "%s.%dof%d.deb",
  151. prefix, curpart, nparts);
  152. }
  153. /* Read data from the original package. */
  154. partrealsize = read(fd_src, partdata, partsize);
  155. if (partrealsize < 0)
  156. ohshite("mksplit: read");
  157. if ((size_t)partrealsize > maxpartsize) {
  158. ohshit("Header is too long, making part too long. "
  159. "Your package name or version\n"
  160. "numbers must be extraordinarily long, "
  161. "or something. Giving up.\n");
  162. }
  163. /* Split the data. */
  164. fd_dst = open(file_dst.buf, O_CREAT | O_WRONLY, 0644);
  165. if (fd_dst < 0)
  166. ohshite(_("unable to open file '%s'"), file_dst.buf);
  167. /* Write the ar header. */
  168. dpkg_ar_put_magic(file_dst.buf, fd_dst);
  169. /* Write the debian-split part. */
  170. varbufprintf(&partmagic, "%s\n%s\n%s\n%s\n%zu\n%zu\n%d/%d\n",
  171. SPLITVERSION, package, version, hash,
  172. st.st_size, partsize, curpart, nparts);
  173. dpkg_ar_member_put_mem(file_dst.buf, fd_dst, PARTMAGIC,
  174. partmagic.buf, partmagic.used);
  175. varbufreset(&partmagic);
  176. /* Write the data part. */
  177. varbufprintf(&partname, "data.%d", curpart);
  178. dpkg_ar_member_put_mem(file_dst.buf, fd_dst, partname.buf,
  179. partdata, (size_t)partrealsize);
  180. varbufreset(&partname);
  181. close(fd_dst);
  182. printf("%d ", curpart);
  183. curpart++;
  184. }
  185. varbuf_destroy(&file_dst);
  186. varbuf_destroy(&partname);
  187. varbuf_destroy(&partmagic);
  188. free(partdata);
  189. free(prefixdir);
  190. free(msdos_prefix);
  191. close(fd_src);
  192. printf("done\n");
  193. return 0;
  194. }
  195. void do_split(const char *const *argv) {
  196. const char *sourcefile, *prefix;
  197. char *palloc;
  198. int l;
  199. size_t partsize;
  200. sourcefile= *argv++;
  201. if (!sourcefile)
  202. badusage(_("--split needs a source filename argument"));
  203. prefix= *argv++;
  204. if (prefix && *argv)
  205. badusage(_("--split takes at most a source filename and destination prefix"));
  206. if (!prefix) {
  207. l= strlen(sourcefile);
  208. palloc= nfmalloc(l+1);
  209. strcpy(palloc,sourcefile);
  210. if (!strcmp(palloc+l-(sizeof(DEBEXT)-1),DEBEXT)) {
  211. l -= (sizeof(DEBEXT)-1);
  212. palloc[l] = '\0';
  213. }
  214. prefix= palloc;
  215. }
  216. partsize = maxpartsize - HEADERALLOWANCE;
  217. mksplit(sourcefile, prefix, partsize, maxpartsize, msdos);
  218. exit(0);
  219. }