#include <limits.h>
#include <regex.h>
#include <signal.h>
#define ERR (-2)
#define EMOD (-3)
#define FATAL (-4)
#define MINBUFSZ 512
#define SE_MAX 30
#define LINECHARS INT_MAX
#define GLB 001
#define GPR 002
#define GLS 004
#define GNP 010
#define GSG 020
typedef struct line {
struct line *q_forw;
struct line *q_back;
off_t seek;
int len;
} line_t;
typedef struct undo {
#define UADD 0
#define UDEL 1
#define UMOV 2
#define VMOV 3
int type;
line_t *h;
line_t *t;
} undo_t;
#ifndef max
# define max(a,b) ((a) > (b) ? (a) : (b))
#endif
#ifndef min
# define min(a,b) ((a) < (b) ? (a) : (b))
#endif
#define INC_MOD(l, k) ((l) + 1 > (k) ? 0 : (l) + 1)
#define DEC_MOD(l, k) ((l) - 1 < 0 ? (k) : (l) - 1)
#define SPL1() mutex++
#define SPL0() \
do { \
if (--mutex == 0) { \
if (sigint) \
handle_int(SIGINT); \
} \
} while (0)
#define STRTOI(i, p) { \
long l = strtol(p, &p, 10); \
if (l <= INT_MIN || l >= INT_MAX) { \
seterrmsg("number out of range"); \
i = 0; \
return ERR; \
} else \
i = (int)l; \
}
#define REALLOC(b,n,i,err) \
if ((i) > (n)) { \
int ti = (n); \
char *ts; \
SPL1(); \
if ((ts = realloc((b), ti += max((i), MINBUFSZ))) == NULL) { \
perror(NULL); \
seterrmsg("out of memory"); \
SPL0(); \
return err; \
} \
(n) = ti; \
(b) = ts; \
SPL0(); \
}
#define REQUE(pred, succ) (pred)->q_forw = (succ), (succ)->q_back = (pred)
#define INSQUE(elem, pred) \
{ \
REQUE((elem), (pred)->q_forw); \
REQUE((pred), elem); \
}
#define REMQUE(elem) REQUE((elem)->q_back, (elem)->q_forw);
#define NUL_TO_NEWLINE(s, l) translit_text(s, l, '\0', '\n')
#define NEWLINE_TO_NUL(s, l) translit_text(s, l, '\n', '\0')
void add_line_node(line_t *);
int build_active_list(int);
void clear_active_list(void);
void clear_undo_stack(void);
int close_sbuf(void);
int delete_lines(int, int);
int display_lines(int, int, int);
int exec_command(void);
int exec_global(int, int);
int extract_addr_range(void);
int extract_subst_tail(int *, int *);
line_t *get_addressed_line_node(int);
regex_t *get_compiled_pattern(void);
char *get_extended_line(int *, int);
int get_line_node_addr(line_t *);
char *get_sbuf_line(line_t *);
int get_tty_line(void);
void handle_hup(void);
void handle_int(int);
int has_trailing_escape(char *, char *);
void init_buffers(void);
int open_sbuf(void);
int pop_undo_stack(void);
undo_t *push_undo_stack(int, int, int);
char *put_sbuf_line(char *);
int put_tty_line(char *, int, int, int);
void quit(int);
int read_file(char *, int);
int search_and_replace(regex_t *, int, int);
void seterrmsg(char *);
char *strip_escapes(char *);
char *translit_text(char *, int, int, int);
void unmark_line_node(line_t *);
void unset_active_nodes(line_t *, line_t *);
int write_file(char *, char *, int, int);
extern char *ibuf;
extern char *ibufp;
extern int ibufsz;
extern int isbinary;
extern int isglobal;
extern int modified;
extern volatile sig_atomic_t mutex;
extern volatile sig_atomic_t sighup;
extern volatile sig_atomic_t sigint;
extern int addr_last;
extern int current_addr;
extern int first_addr;
extern int lineno;
extern int second_addr;