summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorOtto Moerbeek <otto@cvs.openbsd.org>2003-08-10 21:28:49 +0000
committerOtto Moerbeek <otto@cvs.openbsd.org>2003-08-10 21:28:49 +0000
commit5eb11a7915208c383a6daeccb907fddee5508bff (patch)
tree7426b0bf81f987a35382d1be7fb757145f2a9652
parent1685ecf08523f032e95946f4748a67938a1ded86 (diff)
Do not add an extra newline at the end if the last line of the input
file contains no newline and the diff does not touch the last line. Contributions from millert@. ok millert@ tedu@
-rw-r--r--usr.bin/patch/common.h3
-rw-r--r--usr.bin/patch/inp.c8
-rw-r--r--usr.bin/patch/patch.c39
3 files changed, 31 insertions, 19 deletions
diff --git a/usr.bin/patch/common.h b/usr.bin/patch/common.h
index 9e49c26ad18..b76dc15d072 100644
--- a/usr.bin/patch/common.h
+++ b/usr.bin/patch/common.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: common.h,v 1.22 2003/08/01 20:30:48 otto Exp $ */
+/* $OpenBSD: common.h,v 1.23 2003/08/10 21:28:48 otto Exp $ */
#include <stdbool.h>
@@ -71,6 +71,7 @@ extern bool canonicalize;
/* TRUE if -C was specified on command line. */
extern bool check_only;
extern bool warn_on_invalid_line;
+extern bool last_line_missing_eol;
#define CONTEXT_DIFF 1
diff --git a/usr.bin/patch/inp.c b/usr.bin/patch/inp.c
index dea1e7f3610..87d560b5630 100644
--- a/usr.bin/patch/inp.c
+++ b/usr.bin/patch/inp.c
@@ -1,7 +1,7 @@
-/* $OpenBSD: inp.c,v 1.25 2003/08/08 07:53:19 otto Exp $ */
+/* $OpenBSD: inp.c,v 1.26 2003/08/10 21:28:48 otto Exp $ */
#ifndef lint
-static const char rcsid[] = "$OpenBSD: inp.c,v 1.25 2003/08/08 07:53:19 otto Exp $";
+static const char rcsid[] = "$OpenBSD: inp.c,v 1.26 2003/08/10 21:28:48 otto Exp $";
#endif /* not lint */
#include <sys/types.h>
@@ -252,6 +252,7 @@ plan_a(const char *filename)
}
/* if the last line contains no EOL, append one */
if (i_size > 0 && i_womp[i_size - 1] != '\n') {
+ last_line_missing_eol = true;
/* fix last line */
sz = s - i_ptr[iline];
p = malloc(sz + 1);
@@ -268,7 +269,8 @@ plan_a(const char *filename)
i_ptr[iline] = p;
/* count the extra line and make it point to some valid mem */
i_ptr[++iline] = "";
- }
+ } else
+ last_line_missing_eol = false;
input_lines = iline - 1;
diff --git a/usr.bin/patch/patch.c b/usr.bin/patch/patch.c
index 6f91f1c032f..97e461f75c3 100644
--- a/usr.bin/patch/patch.c
+++ b/usr.bin/patch/patch.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: patch.c,v 1.36 2003/08/10 18:39:43 otto Exp $ */
+/* $OpenBSD: patch.c,v 1.37 2003/08/10 21:28:48 otto Exp $ */
/*
* patch - a program to apply diffs to original files
@@ -27,7 +27,7 @@
*/
#ifndef lint
-static const char rcsid[] = "$OpenBSD: patch.c,v 1.36 2003/08/10 18:39:43 otto Exp $";
+static const char rcsid[] = "$OpenBSD: patch.c,v 1.37 2003/08/10 21:28:48 otto Exp $";
#endif /* not lint */
#include <sys/types.h>
@@ -68,6 +68,7 @@ char *TMPPATNAME;
bool toutkeep = false;
bool trejkeep = false;
bool warn_on_invalid_line;
+bool last_line_missing_eol;
#ifdef DEBUGGING
int debug = 0;
@@ -93,9 +94,9 @@ static void abort_hunk(void);
static void apply_hunk(LINENUM);
static void init_output(const char *);
static void init_reject(const char *);
-static void copy_till(LINENUM);
+static void copy_till(LINENUM, bool);
static void spew_output(void);
-static void dump_line(LINENUM);
+static void dump_line(LINENUM, bool);
static bool patch_match(LINENUM, LINENUM, LINENUM);
static bool similar(const char *, const char *, int);
static __dead void usage(void);
@@ -732,7 +733,7 @@ apply_hunk(LINENUM where)
while (old <= lastline) {
if (pch_char(old) == '-') {
- copy_till(where + old - 1);
+ copy_till(where + old - 1, false);
if (do_defines) {
if (def_state == OUTSIDE) {
fputs(not_defined, ofp);
@@ -748,7 +749,7 @@ apply_hunk(LINENUM where)
} else if (new > pat_end) {
break;
} else if (pch_char(new) == '+') {
- copy_till(where + old - 1);
+ copy_till(where + old - 1, false);
if (do_defines) {
if (def_state == IN_IFNDEF) {
fputs(else_defined, ofp);
@@ -770,7 +771,7 @@ apply_hunk(LINENUM where)
#endif
my_exit(2);
} else if (pch_char(new) == '!') {
- copy_till(where + old - 1);
+ copy_till(where + old - 1, false);
if (do_defines) {
fputs(not_defined, ofp);
def_state = IN_IFNDEF;
@@ -802,7 +803,7 @@ apply_hunk(LINENUM where)
}
}
if (new <= pat_end && pch_char(new) == '+') {
- copy_till(where + old - 1);
+ copy_till(where + old - 1, false);
if (do_defines) {
if (def_state == OUTSIDE) {
fputs(if_defined, ofp);
@@ -846,14 +847,20 @@ init_reject(const char *name)
/*
* Copy input file to output, up to wherever hunk is to be applied.
+ * If endoffile is true, treat the last line specially since it may
+ * lack a newline.
*/
static void
-copy_till(LINENUM lastline)
+copy_till(LINENUM lastline, bool endoffile)
{
if (last_frozen_line > lastline)
fatal("misordered hunks! output would be garbled\n");
- while (last_frozen_line < lastline)
- dump_line(++last_frozen_line);
+ while (last_frozen_line < lastline) {
+ if (++last_frozen_line == lastline && endoffile)
+ dump_line(last_frozen_line, !last_line_missing_eol);
+ else
+ dump_line(last_frozen_line, true);
+ }
}
/*
@@ -867,7 +874,7 @@ spew_output(void)
say("il=%ld lfl=%ld\n", input_lines, last_frozen_line);
#endif
if (input_lines)
- copy_till(input_lines); /* dump remainder of file */
+ copy_till(input_lines, true); /* dump remainder of file */
fclose(ofp);
ofp = NULL;
}
@@ -876,7 +883,7 @@ spew_output(void)
* Copy one line from input to output.
*/
static void
-dump_line(LINENUM line)
+dump_line(LINENUM line, bool write_newline)
{
char *s;
@@ -884,8 +891,10 @@ dump_line(LINENUM line)
if (s == NULL)
return;
/* Note: string is not NUL terminated. */
- for (; putc(*s, ofp) != '\n'; s++)
- ;
+ for (; *s != '\n'; s++)
+ putc(*s, ofp);
+ if (write_newline)
+ putc('\n', ofp);
}
/*