/*
 * regsub @(#)regsub.c	1.3 of 2 April 86
 *
 * Copyright (c) 1986 by University of Toronto. Written by Henry Spencer.  Not
 * derived from licensed software.
 *
 * Permission is granted to anyone to use this software for any purpose on any
 * computer system, and to redistribute it freely, subject to the following
 * restrictions:
 *
 * 1. The author is not responsible for the consequences of use of this
 * software, no matter how awful, even if they arise from defects in it.
 *
 * 2. The origin of this software must not be misrepresented, either by explicit
 * claim or by omission.
 *
 * 3. Altered versions must be plainly marked as such, and must not be
 * misrepresented as being the original software.
 */

#define _GNU_SOURCE

#define rep_NEED_REGEXP_INTERNALS
#include "repint.h"

#include <stdio.h>
#include <string.h>

/*
 * CHANGED, 14-Jan-93, by J.Harper,
 * added #ifdef __STDC__ prototype sections so I can use registerized
 * arguments
 *
 * also, I added the regsublen() function for safety & general usefulness
 * (regsub() has no checks for overstepping its dest string)
 */

#ifndef CHARBITS
#define UCHARAT(p)	((int)*(unsigned char *)(p))
#else
#define UCHARAT(p)	((int)*(p)&CHARBITS)
#endif

/*
 * - regsub - perform substitutions after a regexp match
 *
 * data is null if the last match was a string, or the TX if the last
 * match was on a buffer.
 */
void
rep_default_regsub(lasttype, matches, source, dest, data)
    int		    lasttype;
    rep_regsubs	   *matches;
    char	   *source;
    char	   *dest;
    void	   *data;
{
    register char  *src;
    register char  *dst;
    register char   c;
    register int    no;
    register int    len;

    if (matches == NULL || source == NULL || dest == NULL) {
	rep_regerror("NULL parm to regsub");
	return;
    }
    if ((lasttype == rep_reg_string && !rep_STRINGP(rep_VAL(data)))
	|| (lasttype == rep_reg_obj))
    {
	rep_regerror("Bad type of data to regsub");
	return;
    }
    src = source;
    dst = dest;
    while ((c = *src++) != '\0')
    {
	if (c == '&')
	    no = 0;
	else if (c == '\\' && '0' <= *src && *src <= '9')
	    no = *src++ - '0';
	else
	    no = -1;

	if (no < 0) {		/* Ordinary character. */
	    if (c == '\\' && (*src == '\\' || *src == '&'))
		c = *src++;
	    *dst++ = c;
	} else {
	    if(lasttype == rep_reg_string)
	    {
		if (matches->string.startp[no] != NULL
		    && matches->string.endp[no] != NULL)
		{
		    len = matches->string.endp[no]
			  - matches->string.startp[no];
		    (void) strncpy(dst, matches->string.startp[no], len);
		    dst += len;
		    if (len != 0 && *(dst - 1) == '\0')
		    {
			/* strncpy hit NUL. */
			rep_regerror("damaged match string");
			return;
		    }
		}
	    }
	}
    }
    *dst++ = '\0';
}

/*
 * - regsublen - dummy regsub() returning length of contructed string,
 * including terminating '\0'
 */
int
rep_default_regsublen(lasttype, matches, source, data)
    int		    lasttype;
    rep_regsubs	   *matches;
    char	   *source;
    void	   *data;
{
    register char  *src;
    register char   c;
    register int    no;
    register int    dstlen = 1;

    if (matches == NULL || source == NULL) {
	rep_regerror("NULL parm to regsublen");
	return(0);
    }
    if ((lasttype == rep_reg_string && !rep_STRINGP(rep_VAL(data)))
	|| (lasttype == rep_reg_obj))
    {
	rep_regerror("Bad type of data to regsublen");
	return (0);
    }
    src = source;
    while ((c = *src++) != '\0') {
	if (c == '&')
	    no = 0;
	else if (c == '\\' && '0' <= *src && *src <= '9')
	    no = *src++ - '0';
	else
	    no = -1;

	if (no < 0) {		/* Ordinary character. */
	    if (c == '\\' && (*src == '\\' || *src == '&'))
		c = *src++;
	    dstlen++;
	} else {
	    if(lasttype == rep_reg_string)
	    {
		if (matches->string.startp[no] != NULL
		    && matches->string.endp[no] != NULL)
		{
		    dstlen += matches->string.endp[no]
			      - matches->string.startp[no];
		}
	    }
	}
    }
    return(dstlen);
}


syntax highlighted by Code2HTML, v. 0.9.1