*
* Copyright (C) 2006 by Rob Sullivan, with ideas from code by Walter Harms
*
- * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. */
+ * Licensed under GPLv2 or later, see file LICENSE in this source tree.
+ */
+//config:config CKSUM
+//config: bool "cksum (4.1 kb)"
+//config: default y
+//config: help
+//config: cksum is used to calculate the CRC32 checksum of a file.
+
+//applet:IF_CKSUM(APPLET_NOEXEC(cksum, cksum, BB_DIR_USR_BIN, BB_SUID_DROP, cksum))
+/* bb_common_bufsiz1 usage here is safe wrt NOEXEC: not expecting it to be zeroed. */
+
+//kbuild:lib-$(CONFIG_CKSUM) += cksum.o
+
+//usage:#define cksum_trivial_usage
+//usage: "FILE..."
+//usage:#define cksum_full_usage "\n\n"
+//usage: "Calculate the CRC32 checksums of FILEs"
#include "libbb.h"
+#include "common_bufsiz.h"
+
+/* This is a NOEXEC applet. Be very careful! */
int cksum_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
-int cksum_main(int argc ATTRIBUTE_UNUSED, char **argv)
+int cksum_main(int argc UNUSED_PARAM, char **argv)
{
uint32_t *crc32_table = crc32_filltable(NULL, 1);
- uint32_t crc;
- long length, filesize;
- int bytes_read;
- uint8_t *cp;
+ int exit_code = EXIT_SUCCESS;
#if ENABLE_DESKTOP
getopt32(argv, ""); /* coreutils 6.9 compat */
argv++;
#endif
+ setup_common_bufsiz();
do {
+ uint32_t crc;
+ off_t filesize;
int fd = open_or_warn_stdin(*argv ? *argv : bb_msg_standard_input);
- if (fd < 0)
+ if (fd < 0) {
+ exit_code = EXIT_FAILURE;
continue;
- crc = 0;
- length = 0;
+ }
+ crc = 0;
+ filesize = 0;
#define read_buf bb_common_bufsiz1
- while ((bytes_read = safe_read(fd, read_buf, sizeof(read_buf))) > 0) {
- cp = read_buf;
- length += bytes_read;
- do {
- crc = (crc << 8) ^ crc32_table[(crc >> 24) ^ *cp++];
- } while (--bytes_read);
+ for (;;) {
+ uoff_t t;
+ int bytes_read = safe_read(fd, read_buf, COMMON_BUFSIZE);
+ if (bytes_read > 0) {
+ filesize += bytes_read;
+ } else {
+ /* Checksum filesize bytes, LSB first, and exit */
+ close(fd);
+ fd = -1; /* break flag */
+ t = filesize;
+ bytes_read = 0;
+ while (t != 0) {
+ read_buf[bytes_read++] = (uint8_t)t;
+ t >>= 8;
+ }
+ }
+ crc = crc32_block_endian1(crc, read_buf, bytes_read, crc32_table);
+ if (fd < 0)
+ break;
}
- close(fd);
-
- filesize = length;
-
- for (; length; length >>= 8)
- crc = (crc << 8) ^ crc32_table[((crc >> 24) ^ length) & 0xff];
- crc ^= 0xffffffffL;
- printf((*argv ? "%" PRIu32 " %li %s\n" : "%" PRIu32 " %li\n"),
- crc, filesize, *argv);
+ crc = ~crc;
+ printf((*argv ? "%u %"OFF_FMT"u %s\n" : "%u %"OFF_FMT"u\n"),
+ (unsigned)crc, filesize, *argv);
} while (*argv && *++argv);
- fflush_stdout_and_exit(EXIT_SUCCESS);
+ fflush_stdout_and_exit(exit_code);
}