diff options
author | initial.commit <initial.commit@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-07-27 00:38:33 +0000 |
---|---|---|
committer | initial.commit <initial.commit@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-07-27 00:38:33 +0000 |
commit | 584cd5cbd7be997400ccb8db24ae5410b0b88117 (patch) | |
tree | 083e5f1f48d019e0f07b96fef7179483df53c823 /third_party/bspatch/mbspatch.cc | |
parent | f5b16fed647e941aa66933178da85db2860d639b (diff) | |
download | chromium_src-584cd5cbd7be997400ccb8db24ae5410b0b88117.zip chromium_src-584cd5cbd7be997400ccb8db24ae5410b0b88117.tar.gz chromium_src-584cd5cbd7be997400ccb8db24ae5410b0b88117.tar.bz2 |
Add third_party to the repository.
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@19 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'third_party/bspatch/mbspatch.cc')
-rw-r--r-- | third_party/bspatch/mbspatch.cc | 234 |
1 files changed, 234 insertions, 0 deletions
diff --git a/third_party/bspatch/mbspatch.cc b/third_party/bspatch/mbspatch.cc new file mode 100644 index 0000000..2b4d76a --- /dev/null +++ b/third_party/bspatch/mbspatch.cc @@ -0,0 +1,234 @@ +/*- + * Copyright 2003,2004 Colin Percival + * All rights reserved + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted providing that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR + * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY + * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, + * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING + * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + * Changelog: + * 2005-04-26 - Define the header as a C structure, add a CRC32 checksum to + * the header, and make all the types 32-bit. + * --Benjamin Smedberg <benjamin@smedbergs.us> + * 2007-11-14 - Added CalculateCrc() and ApplyBinaryPatch() methods. + * --Rahul Kuchhal + */ + +#include "mbspatch.h" + +#include <sys/stat.h> +#include <stdlib.h> +#include <stdio.h> +#include <fcntl.h> +#include <string.h> +#include <limits.h> + +#ifdef _WIN32 +# include <io.h> +# include <winsock2.h> +#else +# include <unistd.h> +# include <arpa/inet.h> +#endif + +extern "C" { +#include <7zCrc.h> +} + +#ifndef SSIZE_MAX +# define SSIZE_MAX LONG_MAX +#endif + +int +MBS_ReadHeader(int fd, MBSPatchHeader *header) +{ + int s = read(fd, header, sizeof(MBSPatchHeader)); + if (s != sizeof(MBSPatchHeader)) + return READ_ERROR; + + header->slen = ntohl(header->slen); + header->scrc32 = ntohl(header->scrc32); + header->dlen = ntohl(header->dlen); + header->cblen = ntohl(header->cblen); + header->difflen = ntohl(header->difflen); + header->extralen = ntohl(header->extralen); + + struct stat hs; + s = fstat(fd, &hs); + if (s) + return READ_ERROR; + + if (memcmp(header->tag, "MBDIFF10", 8) != 0) + return UNEXPECTED_ERROR; + + if (sizeof(MBSPatchHeader) + + header->cblen + + header->difflen + + header->extralen != int(hs.st_size)) + return UNEXPECTED_ERROR; + + return OK; +} + +int +MBS_ApplyPatch(const MBSPatchHeader *header, int patchfd, + unsigned char *fbuffer, int filefd) +{ + unsigned char *fbufend = fbuffer + header->slen; + + unsigned char *buf = (unsigned char*) malloc(header->cblen + + header->difflen + + header->extralen); + if (!buf) + return MEM_ERROR; + + int rv = OK; + + int r = header->cblen + header->difflen + header->extralen; + unsigned char *wb = buf; + while (r) { + int c = read(patchfd, wb, (r > SSIZE_MAX) ? SSIZE_MAX : r); + if (c < 0) { + rv = READ_ERROR; + goto end; + } + + r -= c; + + if (c == 0 && r) { + rv = UNEXPECTED_ERROR; + goto end; + } + } + + { + MBSPatchTriple *ctrlsrc = (MBSPatchTriple*) buf; + unsigned char *diffsrc = buf + header->cblen; + unsigned char *extrasrc = diffsrc + header->difflen; + + MBSPatchTriple *ctrlend = (MBSPatchTriple*) diffsrc; + unsigned char *diffend = extrasrc; + unsigned char *extraend = extrasrc + header->extralen; + + do { + ctrlsrc->x = ntohl(ctrlsrc->x); + ctrlsrc->y = ntohl(ctrlsrc->y); + ctrlsrc->z = ntohl(ctrlsrc->z); + +#ifdef DEBUG_bsmedberg + printf("Applying block:\n" + " x: %u\n" + " y: %u\n" + " z: %i\n", + ctrlsrc->x, + ctrlsrc->y, + ctrlsrc->z); +#endif + + /* Add x bytes from oldfile to x bytes from the diff block */ + + if (fbuffer + ctrlsrc->x > fbufend || + diffsrc + ctrlsrc->x > diffend) { + rv = UNEXPECTED_ERROR; + goto end; + } + for (unsigned int i = 0; i < ctrlsrc->x; ++i) { + diffsrc[i] += fbuffer[i]; + } + if ((int) write(filefd, diffsrc, ctrlsrc->x) != ctrlsrc->x) { + rv = WRITE_ERROR; + goto end; + } + fbuffer += ctrlsrc->x; + diffsrc += ctrlsrc->x; + + /* Copy y bytes from the extra block */ + + if (extrasrc + ctrlsrc->y > extraend) { + rv = UNEXPECTED_ERROR; + goto end; + } + if ((int) write(filefd, extrasrc, ctrlsrc->y) != ctrlsrc->y) { + rv = WRITE_ERROR; + goto end; + } + extrasrc += ctrlsrc->y; + + /* "seek" forwards in oldfile by z bytes */ + + if (fbuffer + ctrlsrc->z > fbufend) { + rv = UNEXPECTED_ERROR; + goto end; + } + fbuffer += ctrlsrc->z; + + /* and on to the next control block */ + + ++ctrlsrc; + } while (ctrlsrc < ctrlend); + } + +end: + free(buf); + return rv; +} + +int CalculateCrc(const unsigned char *buf, int size) { + CrcGenerateTable(); + unsigned int crc = 0xffffffffL; + crc = ~CrcCalc(buf, size); + return crc; +} + +int ApplyBinaryPatch(const char *old_file, const char *patch_file, + const char *new_file) { + int ret = 0; + int pfd = open(patch_file, O_RDONLY | _O_BINARY); + if (pfd < 0) return READ_ERROR; + + MBSPatchHeader header; + if (ret = MBS_ReadHeader(pfd, &header)) return ret; + + int ofd = open(old_file, O_RDONLY | _O_BINARY); + if (ofd < 0) return READ_ERROR; + + struct stat os; + if (ret = fstat(ofd, &os)) return READ_ERROR; + if (os.st_size != header.slen) return UNEXPECTED_ERROR; + + unsigned char *buf = (unsigned char*) malloc(header.slen); + if (!buf) return MEM_ERROR; + + if (read(ofd, buf, header.slen) != header.slen) return READ_ERROR; + if (CalculateCrc(buf, header.slen) != header.scrc32) + return CRC_ERROR; + + int nfd = open(new_file, O_WRONLY | O_TRUNC | O_CREAT | _O_BINARY); + if (nfd < 0) return READ_ERROR; + + MBS_ApplyPatch(&header, pfd, buf, nfd); + + free(buf); + close(pfd); + close(ofd); + close(nfd); + return OK; +} + |