Compare commits
13 Commits
8c33f0116c
...
0.13.17
| Author | SHA1 | Date | |
|---|---|---|---|
|
bf06e91be5
|
|||
|
5d5a6d2172
|
|||
|
691e94c0c1
|
|||
|
cf744efc1b
|
|||
|
bab3cdd90e
|
|||
|
abfe7046e7
|
|||
|
6ed7089b25
|
|||
|
571796fe0d
|
|||
|
9e8b82c4bb
|
|||
|
906eb92f5a
|
|||
|
9f420131ee
|
|||
|
1fab60d779
|
|||
|
fe175cab19
|
22
docs/dj.1
22
docs/dj.1
@@ -41,10 +41,6 @@ with a skip offset of 1 skips one byte into the input and reads from the second
|
||||
byte onwards. A programmer may think of a file as a zero-indexed array of
|
||||
bytes; in this analogy, the offset given is the index of the byte at which to
|
||||
start reading or writing.
|
||||
|
||||
Seeks and skips aren\(cqt counted in the output statistics because they're
|
||||
guaranteed to succeed (or the utility will exit unsuccessfully, before it has
|
||||
written any data).
|
||||
.\"
|
||||
.SH OPTIONS
|
||||
|
||||
@@ -185,9 +181,20 @@ option is specified, this could make written data nonsensical.
|
||||
|
||||
Existing files are not truncated on ouput and are instead overwritten.
|
||||
|
||||
Many lowercase options have capitalized variants and vice-versa which can be
|
||||
confusing. Capitalized options tend to affect output or are more intense
|
||||
versions of lowercase options.
|
||||
The options
|
||||
.B -b
|
||||
and
|
||||
.B -B
|
||||
could be confused for each other, and so could
|
||||
.B -s
|
||||
and
|
||||
.BR -S .
|
||||
The lowercase option affects input and the capitalized option affects output.
|
||||
|
||||
The skipped or sought bytes while processing irregular files, such as streams,
|
||||
are reported in the diagnostic output, because they were actually read or
|
||||
written. This is as opposed to bytes skipped while processing regular files,
|
||||
which are not reported.
|
||||
.\"
|
||||
.SH RATIONALE
|
||||
|
||||
@@ -208,3 +215,4 @@ Copyright \(co 2023 DTB. License AGPLv3+: GNU AGPL version 3 or later
|
||||
.\"
|
||||
.SH SEE ALSO
|
||||
.BR dd (1p)
|
||||
.BR lseek (3p)
|
||||
|
||||
168
src/dj.c
168
src/dj.c
@@ -37,17 +37,18 @@ char *program_name = "dj";
|
||||
* writing ends of its jockeyed "pipe". User-configurable members are noted
|
||||
* with their relevant options. */
|
||||
struct Io{
|
||||
int bs; /* buffer size (-bB) */
|
||||
char *buf; /* buffer */
|
||||
char *fn; /* file name (-io) */
|
||||
size_t bs; /* buffer size (-bB) */
|
||||
size_t bufuse; /* buffer usage */
|
||||
char *buf; /* buffer */
|
||||
size_t bytes; /* bytes processed */
|
||||
int fd; /* file descriptor */
|
||||
int fl; /* file opening flags */
|
||||
char *fn; /* file name (may be stdin_name or stdout_name) (-io) */
|
||||
size_t prec; /* partial records processed */
|
||||
size_t rec; /* records processed */
|
||||
long seek; /* bytes to seek/skip (will be 0 after skippage) (-sS) */
|
||||
};
|
||||
size_t bytes; /* bytes processed */
|
||||
size_t prec; /* partial records processed */
|
||||
size_t rec; /* records processed */
|
||||
long seek; /* remaining bytes to seek/skip (-sS) */
|
||||
int error; /* errno */
|
||||
int fd; /* file descriptor */
|
||||
int fl; /* file opening flags */
|
||||
};
|
||||
|
||||
/* To be assigned to main:fmt and used with printio(). */
|
||||
static char *fmt_asv = "%d\037%d\036%d\037%d\035%d\036%d\034";
|
||||
@@ -56,7 +57,6 @@ static char *fmt_human = "%d+%d > %d+%d; %d > %d\n";
|
||||
static char *stdin_name = "<stdin>";
|
||||
static char *stdout_name = "<stdout>";
|
||||
|
||||
|
||||
static int creat_mode = S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH
|
||||
| S_IWOTH; /* Consistent with touch(1p). */
|
||||
static int read_flags = O_RDONLY; /* Consistent with Busybox dd(1). */
|
||||
@@ -69,16 +69,23 @@ static int write_flags = O_WRONLY | O_CREAT;
|
||||
|
||||
static struct Io *
|
||||
Io_read(struct Io *io){
|
||||
int t;
|
||||
|
||||
assert(io->bs > 0);
|
||||
assert(io->bufuse < io->bs);
|
||||
|
||||
io->bytes += (io->bufuse = read(io->fd, io->buf, io->bs));
|
||||
if((t = read(io->fd, &(io->buf)[io->bufuse], io->bs - io->bufuse)) < 0){
|
||||
io->error = errno;
|
||||
t = 0;
|
||||
}
|
||||
|
||||
io->bufuse += t;
|
||||
io->bytes += t;
|
||||
io->prec += (0 < io->bufuse && io->bufuse < io->bs);
|
||||
io->rec += (io->bufuse == io->bs);
|
||||
|
||||
assert(io->bufuse <= io->bs);
|
||||
|
||||
io->prec += (0 < io->bufuse && io->bufuse < io->bs);
|
||||
io->rec += (io->bufuse == io->bs);
|
||||
|
||||
return io;
|
||||
}
|
||||
|
||||
@@ -89,8 +96,11 @@ Io_write(struct Io *io){
|
||||
assert(io->bufuse > 0);
|
||||
assert(io->bufuse <= io->bs);
|
||||
|
||||
if((t = write(io->fd, io->buf, io->bufuse)) > 0)
|
||||
memmove(io->buf, io->buf + t, (io->bufuse -= t));
|
||||
if((t = write(io->fd, io->buf, io->bufuse)) < 0){
|
||||
io->error = errno;
|
||||
t = 0;
|
||||
}else if(t > 0)
|
||||
memmove(io->buf, &(io->buf)[t], (io->bufuse -= t));
|
||||
|
||||
io->bytes += t;
|
||||
io->prec += (t > 0 && io->bufuse > 0);
|
||||
@@ -99,13 +109,9 @@ Io_write(struct Io *io){
|
||||
return io;
|
||||
}
|
||||
|
||||
/* Prints an error message suitable for the event of an operating system error,
|
||||
* with the error itself to be described in the string s. */
|
||||
static int
|
||||
oserr(char *s){
|
||||
|
||||
fprintf(stderr, "%s: %s: %s\n", program_name, s, strerror(errno));
|
||||
|
||||
oserr(char *e, int n){
|
||||
fprintf(stderr, "%s: %s: %s\n", program_name, e, strerror(n));
|
||||
return EX_OSERR;
|
||||
}
|
||||
|
||||
@@ -152,20 +158,21 @@ int main(int argc, char *argv[]){
|
||||
char *fmt; /* == fmt_asv (default) or fmt_human (-H) */
|
||||
size_t i; /* side of io being modified */
|
||||
char noerror; /* 0=exits (default) 1=retries on partial reads or writes */
|
||||
struct Io io[2];
|
||||
struct Io io[2 /* { in, out } */];
|
||||
|
||||
/* Set defaults. */
|
||||
align = -1;
|
||||
count = 0;
|
||||
fmt = fmt_asv;
|
||||
noerror = 0;
|
||||
for(i = 0; i < 2; ++i){
|
||||
for(i = 0; i < (sizeof io) / (sizeof *io); ++i){
|
||||
io[i].bs = 1024 /* 1 KiB */; /* GNU dd(1) default; POSIX says 512B */
|
||||
io[i].bufuse = 0;
|
||||
io[i].bytes = 0;
|
||||
io[i].fd = i == 0 ? STDIN_FILENO : STDOUT_FILENO;
|
||||
io[i].fn = i == 0 ? stdin_name : stdout_name;
|
||||
io[i].fl = i == 0 ? read_flags : write_flags;
|
||||
io[i].error = 0;
|
||||
io[i].prec = 0;
|
||||
io[i].rec = 0;
|
||||
io[i].seek = 0;
|
||||
@@ -192,7 +199,7 @@ int main(int argc, char *argv[]){
|
||||
break;
|
||||
}
|
||||
}
|
||||
return oserr(optarg);
|
||||
return oserr(optarg, errno);
|
||||
case 'n': noerror = 1; break;
|
||||
case 'H': fmt = fmt_human; break;
|
||||
case 'a':
|
||||
@@ -221,10 +228,10 @@ int main(int argc, char *argv[]){
|
||||
if(argc > optind)
|
||||
return usage(program_name);
|
||||
|
||||
for(i = 0; i < 2; ++i){
|
||||
for(i = 0; i < (sizeof io) / (sizeof *io); ++i){
|
||||
/* buffer allocation */
|
||||
if((io[i].buf = malloc(io[i].bs * (sizeof *(io[i].buf)))) == NULL){
|
||||
fprintf(stderr, "%s: Failed to allocate %d bytes\n",
|
||||
fprintf(stderr, "%s: Failed to allocate %zd bytes\n",
|
||||
program_name, io[i].bs);
|
||||
return EX_OSERR;
|
||||
}
|
||||
@@ -233,58 +240,60 @@ int main(int argc, char *argv[]){
|
||||
io[i].seek = 0;
|
||||
}
|
||||
|
||||
/* hard skipping */
|
||||
if(io[0].seek > 0){
|
||||
do{
|
||||
if((io[0].bufuse = read(
|
||||
io[0].fd, io[0].buf, MIN(io[0].bs, io[0].seek)))
|
||||
== 0)
|
||||
/* second chance */
|
||||
io->bufuse = read(
|
||||
io[0].fd, io[0].buf, MIN(io[0].bs, io[0].seek));
|
||||
}while((io[0].seek -= io[0].bufuse) > 0 && io[0].bufuse != 0);
|
||||
io[0].bufuse = 0;
|
||||
}
|
||||
|
||||
/* hard seeking */
|
||||
if(io[1].seek > 0){
|
||||
memset(io[1].buf, '\0', io[1].bs);
|
||||
/* We're going to cheat and use bufuse as the retval for write(2),
|
||||
* which is fine because it'll be zeroed as this function returns
|
||||
* anyway. */
|
||||
size_t t;
|
||||
do{
|
||||
if((io[1].bufuse = write(
|
||||
io[1].fd, io[1].buf, MIN(io[1].bs, io[1].seek)))
|
||||
== 0)
|
||||
/* second chance */
|
||||
io[1].bufuse = write(
|
||||
io[1].fd, io[1].buf, MIN(io[1].bs, io[1].seek));
|
||||
}while((io[1].seek -= io[1].bufuse) > 0 && io[1].bufuse != 0);
|
||||
memset(io[1].buf, '\0',
|
||||
(t = io[1].bufuse = MIN(io[1].bs, io[1].seek)));
|
||||
if(Io_write(&io[1])->bufuse == t && !noerror && io[1].error == 0)
|
||||
Io_write(&io[1]); /* second chance */
|
||||
if(io[1].error != 0)
|
||||
return oserr(io[1].fn, io[1].error);
|
||||
}while((io[1].seek -= (t - io[1].bufuse)) > 0 && io[1].bufuse != t);
|
||||
io[1].bufuse = 0;
|
||||
}
|
||||
|
||||
/* Sought bytes aren't counted in the statistics because successful seeking
|
||||
* is guaranteed here. */
|
||||
for(i = 0; i < 2; ++i)
|
||||
if(io[i].seek > 0)
|
||||
return oserr(io[i].fn);
|
||||
if(io[1].seek > 0){
|
||||
fprintio(stderr, fmt, io);
|
||||
return oserr(io[1].fn, errno);
|
||||
}
|
||||
|
||||
do{ /* read */
|
||||
if(Io_read(&io[0])->bufuse == 0 && !noerror)
|
||||
Io_read(&io[0]); /* second chance */
|
||||
if(io[0].bufuse == 0) /* that's all she wrote */
|
||||
break;
|
||||
do{
|
||||
assert(io[0].bufuse == 0);
|
||||
|
||||
if(io[0].bufuse < io[0].bs){
|
||||
fprintf(stderr, "%s: Partial read:\n\t", program_name);
|
||||
fprintio(stderr, fmt, io);
|
||||
if(!noerror)
|
||||
count = 1;
|
||||
if(align >= 0){
|
||||
/* fill the rest of the ibuf with padding */
|
||||
memset(io[0].buf + io[0].bufuse, align,
|
||||
io[0].bs - io[0].bufuse);
|
||||
io->bufuse = io->bs;
|
||||
{ /* read */
|
||||
char skipping;
|
||||
size_t t;
|
||||
|
||||
/* hack to intentionally get a partial read from Io_read */
|
||||
if((skipping = (io[0].seek > 0)) && io[0].seek < io[0].bs)
|
||||
io[0].bufuse = io[0].bs - io[0].seek;
|
||||
|
||||
t = io[0].bufuse;
|
||||
if(Io_read(&io[0])->bufuse == t && !noerror && io[0].error == 0)
|
||||
Io_read(&io[0]); /* second chance */
|
||||
assert(io[0].bufuse >= t);
|
||||
if(io[0].bufuse == t) /* that's all she wrote */
|
||||
break;
|
||||
|
||||
if(/* t < io[0].bufuse && */ io[0].bufuse < io[0].bs){
|
||||
fprintf(stderr, "%s: Partial read:\n\t", program_name);
|
||||
fprintio(stderr, fmt, io);
|
||||
if(!noerror)
|
||||
count = 1;
|
||||
if(align >= 0){
|
||||
/* fill the rest of the ibuf with padding */
|
||||
memset(&(io[0].buf)[io[0].bufuse], align,
|
||||
io[0].bs - io[0].bufuse);
|
||||
io->bufuse = io->bs;
|
||||
}
|
||||
}
|
||||
|
||||
if(skipping){
|
||||
io[0].bufuse = 0;
|
||||
count += (count != 0);
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -299,16 +308,16 @@ int main(int argc, char *argv[]){
|
||||
memcpy(io[1].buf, io[0].buf,
|
||||
(io[1].bufuse = (n = MIN(io[0].bufuse, io[1].bs))));
|
||||
/* permute the copied units out of ibuf */
|
||||
memmove(io[0].buf, io[0].buf + n, (io[0].bufuse -= n));
|
||||
memmove(io[0].buf, &(io[0].buf)[n], (io[0].bufuse -= n));
|
||||
}else /* if(io[0].bs < io[1].bs) */ {
|
||||
int n;
|
||||
|
||||
/* drain what we can from ibuf */
|
||||
memcpy(io[1].buf + io[1].bufuse, io[0].buf,
|
||||
memcpy(&(io[1].buf)[io[1].bufuse], io[0].buf,
|
||||
(n = MIN(io[0].bufuse, io[1].bs - io[1].bufuse)));
|
||||
io[1].bufuse += n;
|
||||
/* permute out the copied units */
|
||||
memmove(io[0].buf, io[0].buf + n, io[0].bs - n);
|
||||
memmove(io[0].buf, &(io[0].buf)[n], io[0].bs - n);
|
||||
io[0].bufuse -= n;
|
||||
|
||||
if(io[0].bs + io[1].bufuse <= io[1].bs && count != 1)
|
||||
@@ -316,14 +325,15 @@ int main(int argc, char *argv[]){
|
||||
}
|
||||
|
||||
t = io[1].bufuse;
|
||||
if(Io_write(&io[1])->bufuse == t && !noerror)
|
||||
if(Io_write(&io[1])->bufuse == t && !noerror && io[1].error == 0)
|
||||
Io_write(&io[1]); /* second chance */
|
||||
assert(io[1].bufuse <= t);
|
||||
if(io[1].bufuse == t){ /* no more love */
|
||||
count = 1;
|
||||
break;
|
||||
}
|
||||
|
||||
if(0 < io[1].bufuse && io[1].bufuse < t){
|
||||
if(0 < io[1].bufuse /* && io[1].bufuse < t */){
|
||||
fprintf(stderr, "%s: Partial write:\n\t", program_name);
|
||||
fprintio(stderr, fmt, io);
|
||||
if(!noerror)
|
||||
@@ -334,5 +344,9 @@ int main(int argc, char *argv[]){
|
||||
|
||||
fprintio(stderr, fmt, io);
|
||||
|
||||
for(i = 0; i < (sizeof io) / (sizeof *io); ++i)
|
||||
if(io[i].error)
|
||||
return oserr(io[i].fn, io[i].error);
|
||||
|
||||
return EX_OK;
|
||||
}
|
||||
|
||||
@@ -29,8 +29,11 @@ use getopt::GetOpt;
|
||||
extern crate sysexits;
|
||||
use sysexits::{ EX_OK, EX_OSERR, EX_USAGE };
|
||||
|
||||
extern crate strerror;
|
||||
use strerror::StrError;
|
||||
|
||||
fn oserr(s: &str, e: Error) -> ExitCode {
|
||||
eprintln!("{}: {}", s, e);
|
||||
eprintln!("{}: {}", s, e.strerror());
|
||||
ExitCode::from(EX_OSERR as u8)
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user