summaryrefslogtreecommitdiff
path: root/usr.bin/cvs
diff options
context:
space:
mode:
authorJean-Francois Brousseau <jfb@cvs.openbsd.org>2005-04-07 20:50:23 +0000
committerJean-Francois Brousseau <jfb@cvs.openbsd.org>2005-04-07 20:50:23 +0000
commit1c61c41b98f3d69bb0a38abee5da3091b93207d3 (patch)
treea3dddd499f52ec073ec80d14a1a1b10b5b4a50bb /usr.bin/cvs
parentd01df2650a227661a2bfe17504c79f5b9f87fbdf (diff)
Perform proper escaping of the @ character in all RCS strings when
writing the output to the file. ok joris@
Diffstat (limited to 'usr.bin/cvs')
-rw-r--r--usr.bin/cvs/rcs.c125
-rw-r--r--usr.bin/cvs/rcs.h27
2 files changed, 71 insertions, 81 deletions
diff --git a/usr.bin/cvs/rcs.c b/usr.bin/cvs/rcs.c
index df77a45a4ab..f8f76c2e5bb 100644
--- a/usr.bin/cvs/rcs.c
+++ b/usr.bin/cvs/rcs.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: rcs.c,v 1.41 2005/04/07 16:47:11 jfb Exp $ */
+/* $OpenBSD: rcs.c,v 1.42 2005/04/07 20:50:22 jfb Exp $ */
/*
* Copyright (c) 2004 Jean-Francois Brousseau <jfb@openbsd.org>
* All rights reserved.
@@ -84,6 +84,7 @@ struct rcs_pdata {
char *rp_buf;
size_t rp_blen;
char *rp_bufend;
+ size_t rp_tlen;
/* pushback token buffer */
char rp_ptok[128];
@@ -107,7 +108,7 @@ struct rcs_foo {
};
#define RCS_TOKSTR(rfp) ((struct rcs_pdata *)rfp->rf_pdata)->rp_buf
-#define RCS_TOKLEN(rfp) ((struct rcs_pdata *)rfp->rf_pdata)->rp_blen
+#define RCS_TOKLEN(rfp) ((struct rcs_pdata *)rfp->rf_pdata)->rp_tlen
#ifdef notyet
@@ -198,6 +199,7 @@ static int rcs_gettok (RCSFILE *);
static int rcs_pushtok (RCSFILE *, const char *, int);
static int rcs_growbuf (RCSFILE *);
static int rcs_patch_lines (struct rcs_foo *, struct rcs_foo *);
+static int rcs_strprint (const u_char *, size_t, FILE *);
static struct rcs_delta* rcs_findrev (RCSFILE *, RCSNUM *);
static struct rcs_foo* rcs_splitlines (const char *);
@@ -340,8 +342,7 @@ static int
rcs_write(RCSFILE *rfp)
{
FILE *fp;
- char buf[1024], numbuf[64], *cp;
- size_t rlen, len;
+ char buf[1024], numbuf[64];
struct rcs_access *ap;
struct rcs_sym *symp;
struct rcs_delta *rdp;
@@ -389,11 +390,17 @@ rcs_write(RCSFILE *rfp)
fprintf(fp, " strict;");
fputc('\n', fp);
- if (rfp->rf_comment != NULL)
- fprintf(fp, "comment\t@%s@;\n", rfp->rf_comment);
+ if (rfp->rf_comment != NULL) {
+ fputs("comment\t@", fp);
+ rcs_strprint(rfp->rf_comment, strlen(rfp->rf_comment), fp);
+ fputs("@;\n", fp);
+ }
- if (rfp->rf_expand != NULL)
- fprintf(fp, "expand @ %s @;\n", rfp->rf_expand);
+ if (rfp->rf_expand != NULL) {
+ fputs("expand @", fp);
+ rcs_strprint(rfp->rf_expand, strlen(rfp->rf_expand), fp);
+ fputs("@;\n", fp);
+ }
fprintf(fp, "\n\n");
@@ -411,23 +418,20 @@ rcs_write(RCSFILE *rfp)
numbuf, sizeof(numbuf)));
}
- fprintf(fp, "\ndesc\n@%s@\n\n",
- (rfp->rf_desc == NULL) ? "" : rfp->rf_desc);
+ fputs("\ndesc\n@", fp);
+ if (rfp->rf_desc != NULL)
+ rcs_strprint(rfp->rf_desc, strlen(rfp->rf_desc), fp);
+ fputs("@\n\n", fp);
/* deltatexts */
TAILQ_FOREACH(rdp, &(rfp->rf_delta), rd_list) {
fprintf(fp, "\n%s\n", rcsnum_tostr(rdp->rd_num, numbuf,
sizeof(numbuf)));
- fprintf(fp, "log\n@%s@\ntext\n@", rdp->rd_log);
-
- cp = rdp->rd_text;
- do {
- len = sizeof(buf);
- rlen = rcs_stresc(1, cp, buf, &len);
- fprintf(fp, "%s", buf);
- cp += rlen;
- } while (len != 0);
- fprintf(fp, "@\n\n");
+ fputs("log\n@", fp);
+ rcs_strprint(rdp->rd_log, strlen(rdp->rd_log), fp);
+ fputs("@\ntext\n@", fp);
+ rcs_strprint(rdp->rd_text, rdp->rd_tlen, fp);
+ fputs("@\n\n", fp);
}
fclose(fp);
@@ -1721,6 +1725,7 @@ rcs_parse_deltatext(RCSFILE *rfp)
cvs_log(LP_ERRNO, "failed to copy RCS delta text");
return (-1);
}
+ rdp->rd_tlen = RCS_TOKLEN(rfp);
return (1);
}
@@ -2022,6 +2027,7 @@ rcs_gettok(RCSFILE *rfp)
type = RCS_TOK_ERR;
bp = pdp->rp_buf;
+ pdp->rp_tlen = 0;
*bp = '\0';
if (pdp->rp_pttype != RCS_TOK_ERR) {
@@ -2055,6 +2061,7 @@ rcs_gettok(RCSFILE *rfp)
break;
}
*(bp++) = ch;
+ pdp->rp_tlen++;
if (bp == pdp->rp_bufend - 1) {
len = bp - pdp->rp_buf;
if (rcs_growbuf(rfp) < 0) {
@@ -2090,6 +2097,7 @@ rcs_gettok(RCSFILE *rfp)
pdp->rp_lines++;
*(bp++) = ch;
+ pdp->rp_tlen++;
if (bp == pdp->rp_bufend - 1) {
len = bp - pdp->rp_buf;
if (rcs_growbuf(rfp) < 0) {
@@ -2121,6 +2129,7 @@ rcs_gettok(RCSFILE *rfp)
}
last = ch;
*(bp++) = ch;
+ pdp->rp_tlen++;
}
*bp = '\0';
}
@@ -2146,53 +2155,6 @@ rcs_pushtok(RCSFILE *rfp, const char *tok, int type)
return (0);
}
-/*
- * rcs_stresc()
- *
- * Performs either escaping or unescaping of the string stored in <str>.
- * The operation is to escape special RCS characters if the <esc> argument
- * is 1, or unescape otherwise. The result is stored in the <buf> destination
- * buffer, and <blen> must originally point to the size of <buf>.
- * Returns the number of bytes which have been read from the source <str> and
- * operated on. The <blen> parameter will contain the number of bytes
- * actually copied in <buf>.
- */
-size_t
-rcs_stresc(int esc, const char *str, char *buf, size_t *blen)
-{
- size_t rlen;
- const char *sp;
- char *bp, *bep;
-
- rlen = 0;
- bp = buf;
- bep = buf + *blen - 1;
-
- for (sp = str; (*sp != '\0') && (bp <= (bep - 1)); sp++) {
- if (*sp == '@') {
- if (esc) {
- if (bp > (bep - 2))
- break;
- *(bp++) = '@';
- } else {
- sp++;
- if (*sp != '@') {
- cvs_log(LP_WARN,
- "unknown escape character `%c' in "
- "RCS file", *sp);
- if (*sp == '\0')
- break;
- }
- }
- }
-
- *(bp++) = *sp;
- }
-
- *bp = '\0';
- *blen = (bp - buf);
- return (sp - str);
-}
/*
* rcs_splitlines()
@@ -2297,3 +2259,32 @@ rcs_growbuf(RCSFILE *rf)
return (0);
}
+
+/*
+ * rcs_strprint()
+ *
+ * Output an RCS string <str> of size <slen> to the stream <stream>. Any
+ * '@' characters are escaped. Otherwise, the string can contain arbitrary
+ * binary data.
+ */
+static int
+rcs_strprint(const u_char *str, size_t slen, FILE *stream)
+{
+ const u_char *ap, *ep, *sp;
+ size_t ret;
+
+ ep = str + slen - 1;
+
+ for (sp = str; sp <= ep;) {
+ ap = memchr(sp, '@', ep - sp);
+ if (ap == NULL)
+ ap = ep;
+ ret = fwrite(sp, sizeof(u_char), ap - sp + 1, stream);
+
+ if (*ap == '@')
+ putc('@', stream);
+ sp = ap + 1;
+ }
+
+ return (0);
+}
diff --git a/usr.bin/cvs/rcs.h b/usr.bin/cvs/rcs.h
index 38ba5302c68..b3f4839ca5a 100644
--- a/usr.bin/cvs/rcs.h
+++ b/usr.bin/cvs/rcs.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: rcs.h,v 1.17 2005/04/06 19:12:08 jfb Exp $ */
+/* $OpenBSD: rcs.h,v 1.18 2005/04/07 20:50:22 jfb Exp $ */
/*
* Copyright (c) 2004 Jean-Francois Brousseau <jfb@openbsd.org>
* All rights reserved.
@@ -135,19 +135,19 @@ struct rcs_dlist {
};
struct rcs_delta {
- RCSNUM *rd_num;
- RCSNUM *rd_next;
- u_int rd_flags;
- struct tm rd_date;
- char *rd_author;
- char *rd_state;
- char *rd_log;
- char *rd_text;
-
- struct rcs_dlist rd_snodes;
-
+ RCSNUM *rd_num;
+ RCSNUM *rd_next;
+ u_int rd_flags;
+ struct tm rd_date;
+ char *rd_author;
+ char *rd_state;
+ char *rd_log;
+ char *rd_text;
+ size_t rd_tlen;
+
+ struct rcs_dlist rd_snodes;
TAILQ_HEAD(, rcs_branch) rd_branches;
- TAILQ_ENTRY(rcs_delta) rd_list;
+ TAILQ_ENTRY(rcs_delta) rd_list;
};
@@ -207,7 +207,6 @@ void rcs_kflag_usage (void);
int rcs_kw_expand (RCSFILE *, u_char *, size_t, size_t *);
BUF* rcs_patch (const char *, const char *);
-size_t rcs_stresc (int, const char *, char *, size_t *);
RCSNUM* rcsnum_alloc (void);
RCSNUM* rcsnum_parse (const char *);