1e1bfde1bSBrian Somers /*- 24d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause 31de7b4b8SPedro F. Giffuni * 4e1bfde1bSBrian Somers * Copyright (c) 2005 Brian Somers <brian@FreeBSD.org> 5e1bfde1bSBrian Somers * All rights reserved. 6e1bfde1bSBrian Somers * 7e1bfde1bSBrian Somers * Redistribution and use in source and binary forms, with or without 8e1bfde1bSBrian Somers * modification, are permitted provided that the following conditions 9e1bfde1bSBrian Somers * are met: 10e1bfde1bSBrian Somers * 1. Redistributions of source code must retain the above copyright 11e1bfde1bSBrian Somers * notice, this list of conditions and the following disclaimer. 12e1bfde1bSBrian Somers * 2. Redistributions in binary form must reproduce the above copyright 13e1bfde1bSBrian Somers * notice, this list of conditions and the following disclaimer in the 14e1bfde1bSBrian Somers * documentation and/or other materials provided with the distribution. 15e1bfde1bSBrian Somers * 16e1bfde1bSBrian Somers * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17e1bfde1bSBrian Somers * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18e1bfde1bSBrian Somers * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19e1bfde1bSBrian Somers * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 20e1bfde1bSBrian Somers * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21e1bfde1bSBrian Somers * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22e1bfde1bSBrian Somers * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23e1bfde1bSBrian Somers * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24e1bfde1bSBrian Somers * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25e1bfde1bSBrian Somers * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26e1bfde1bSBrian Somers * SUCH DAMAGE. 27e1bfde1bSBrian Somers */ 28e1bfde1bSBrian Somers 29e1bfde1bSBrian Somers #include <sys/types.h> 30d350e8d7SDag-Erling Smørgrav 31e1bfde1bSBrian Somers #include <err.h> 32e1bfde1bSBrian Somers #include <limits.h> 331f766174SEd Maste #include <stdbool.h> 34e1bfde1bSBrian Somers #include <stdio.h> 35e1bfde1bSBrian Somers #include <stdlib.h> 36e1bfde1bSBrian Somers #include <unistd.h> 37e1bfde1bSBrian Somers 38e1bfde1bSBrian Somers #include "extern.h" 39e1bfde1bSBrian Somers 40*3c37828eSDag-Erling Smørgrav int 414e380e84SKyle Evans c_link(const char *file1, off_t skip1, const char *file2, off_t skip2, 424e380e84SKyle Evans off_t limit) 43e1bfde1bSBrian Somers { 44e1bfde1bSBrian Somers char buf1[PATH_MAX], *p1; 45e1bfde1bSBrian Somers char buf2[PATH_MAX], *p2; 46d350e8d7SDag-Erling Smørgrav ssize_t len1, len2; 47d350e8d7SDag-Erling Smørgrav int dfound; 48e1bfde1bSBrian Somers off_t byte; 49e1bfde1bSBrian Somers u_char ch; 50e1bfde1bSBrian Somers 51e1bfde1bSBrian Somers if ((len1 = readlink(file1, buf1, sizeof(buf1) - 1)) < 0) { 52e1bfde1bSBrian Somers if (!sflag) 53e1bfde1bSBrian Somers err(ERR_EXIT, "%s", file1); 54e1bfde1bSBrian Somers else 55e1bfde1bSBrian Somers exit(ERR_EXIT); 56e1bfde1bSBrian Somers } 57e1bfde1bSBrian Somers 58e1bfde1bSBrian Somers if ((len2 = readlink(file2, buf2, sizeof(buf2) - 1)) < 0) { 59e1bfde1bSBrian Somers if (!sflag) 60e1bfde1bSBrian Somers err(ERR_EXIT, "%s", file2); 61e1bfde1bSBrian Somers else 62e1bfde1bSBrian Somers exit(ERR_EXIT); 63e1bfde1bSBrian Somers } 64e1bfde1bSBrian Somers 65e1bfde1bSBrian Somers if (skip1 > len1) 66e1bfde1bSBrian Somers skip1 = len1; 67e1bfde1bSBrian Somers buf1[len1] = '\0'; 68e1bfde1bSBrian Somers 69e1bfde1bSBrian Somers if (skip2 > len2) 70e1bfde1bSBrian Somers skip2 = len2; 71e1bfde1bSBrian Somers buf2[len2] = '\0'; 72e1bfde1bSBrian Somers 73e1bfde1bSBrian Somers dfound = 0; 74e1bfde1bSBrian Somers byte = 1; 754e380e84SKyle Evans for (p1 = buf1 + skip1, p2 = buf2 + skip2; 764e380e84SKyle Evans *p1 && *p2 && (limit == 0 || byte <= limit); p1++, p2++) { 77e1bfde1bSBrian Somers if ((ch = *p1) != *p2) { 78e1bfde1bSBrian Somers if (xflag) { 79e1bfde1bSBrian Somers dfound = 1; 80e1bfde1bSBrian Somers (void)printf("%08llx %02x %02x\n", 81e1bfde1bSBrian Somers (long long)byte - 1, ch, *p2); 82e1bfde1bSBrian Somers } else if (lflag) { 83e1bfde1bSBrian Somers dfound = 1; 84f66b9b40SKyle Evans if (bflag) 85f66b9b40SKyle Evans (void)printf("%6lld %3o %c %3o %c\n", 86f66b9b40SKyle Evans (long long)byte, ch, ch, *p2, *p2); 87f66b9b40SKyle Evans else 88e1bfde1bSBrian Somers (void)printf("%6lld %3o %3o\n", 89e1bfde1bSBrian Somers (long long)byte, ch, *p2); 90*3c37828eSDag-Erling Smørgrav } else { 91f66b9b40SKyle Evans diffmsg(file1, file2, byte, 1, ch, *p2); 92*3c37828eSDag-Erling Smørgrav return (DIFF_EXIT); 93*3c37828eSDag-Erling Smørgrav } 94e1bfde1bSBrian Somers } 95e1bfde1bSBrian Somers byte++; 96e1bfde1bSBrian Somers } 97e1bfde1bSBrian Somers 98*3c37828eSDag-Erling Smørgrav if (*p1 || *p2) { 99e1bfde1bSBrian Somers eofmsg (*p1 ? file2 : file1); 100*3c37828eSDag-Erling Smørgrav return (DIFF_EXIT); 101*3c37828eSDag-Erling Smørgrav } 102*3c37828eSDag-Erling Smørgrav return (dfound ? DIFF_EXIT : 0); 103e1bfde1bSBrian Somers } 104