#ifndef RCS_H
#define RCS_H
#include <stdio.h>
#include <time.h>
#include "buf.h"
#define RCS_DIFF_DIV \
"==================================================================="
#define RCS_FILE_EXT ",v"
#define RCS_HEAD_BRANCH "HEAD"
#define RCS_HEAD_INIT "1.1"
#define RCS_HEAD_REV ((RCSNUM *)(-1))
#define RCS_CONFLICT_MARKER1 "<<<<<<< "
#define RCS_CONFLICT_MARKER2 ">>>>>>> "
#define RCS_CONFLICT_MARKER3 "=======\n"
#define RCS_SYM_INVALCHAR "$,.:;@"
#define RCS_MAGIC_BRANCH ".0."
#define RCS_STATE_EXP "Exp"
#define RCS_STATE_DEAD "dead"
#define RCS_LOCK_INVAL (-1)
#define RCS_LOCK_LOOSE 0
#define RCS_LOCK_STRICT 1
#define RCS_KW_AUTHOR 0x1000
#define RCS_KW_DATE 0x2000
#define RCS_KW_LOG 0x4000
#define RCS_KW_NAME 0x8000
#define RCS_KW_RCSFILE 0x0100
#define RCS_KW_REVISION 0x0200
#define RCS_KW_SOURCE 0x0400
#define RCS_KW_STATE 0x0800
#define RCS_KW_FULLPATH 0x0010
#define RCS_KW_MDOCDATE 0x0020
#define RCS_KW_LOCKER 0x0040
#define RCS_KW_ID \
(RCS_KW_RCSFILE | RCS_KW_REVISION | RCS_KW_DATE \
| RCS_KW_AUTHOR | RCS_KW_STATE)
#define RCS_KW_HEADER (RCS_KW_ID | RCS_KW_FULLPATH)
#define RCS_KWEXP_NONE 0x01
#define RCS_KWEXP_NAME 0x02
#define RCS_KWEXP_VAL 0x04
#define RCS_KWEXP_LKR 0x08
#define RCS_KWEXP_OLD 0x10
#define RCS_KWEXP_ERR 0x20
#define RCS_KWEXP_DEFAULT (RCS_KWEXP_NAME | RCS_KWEXP_VAL)
#define RCS_KWEXP_KVL (RCS_KWEXP_NAME | RCS_KWEXP_VAL | RCS_KWEXP_LKR)
#define RCS_KWEXP_INVAL(k) \
((k & RCS_KWEXP_ERR) || \
((k & RCS_KWEXP_OLD) && (k & ~RCS_KWEXP_OLD)))
struct rcs_kw {
char kw_str[16];
int kw_type;
};
#define RCS_NKWORDS (sizeof(rcs_expkw)/sizeof(rcs_expkw[0]))
#define RCSNUM_MAXNUM USHRT_MAX
#define RCSNUM_MAXLEN 16
#define RCSNUM_MAXSTR 64
#define RCSNUM_ISBRANCH(n) ((n)->rn_len % 2)
#define RCSNUM_ISBRANCHREV(n) (!((n)->rn_len % 2) && ((n)->rn_len >= 4))
#define RCS_READ (1<<0)
#define RCS_WRITE (1<<1)
#define RCS_CREATE (1<<2)
#define RCS_PARSE_FULLY (1<<3)
#define RCS_PARSED (1<<4)
#define RCS_SYNCED (1<<5)
#define RCS_SLOCK (1<<6)
#define PARSED_DELTAS (1<<7)
#define PARSED_DESC (1<<8)
#define PARSED_DELTATEXTS (1<<9)
#define RCS_RD_DEAD 0x01
#define RCS_RD_SELECT 0x02
#define RCS_COMMITID_MAXLEN 64
typedef struct rcs_num {
u_int rn_len;
u_int16_t rn_id[RCSNUM_MAXLEN];
} RCSNUM;
struct rcs_access {
char *ra_name;
TAILQ_ENTRY(rcs_access) ra_list;
};
struct rcs_sym {
char *rs_name;
RCSNUM *rs_num;
TAILQ_ENTRY(rcs_sym) rs_list;
};
struct rcs_lock {
char *rl_name;
RCSNUM *rl_num;
TAILQ_ENTRY(rcs_lock) rl_list;
};
struct rcs_branch {
RCSNUM *rb_num;
TAILQ_ENTRY(rcs_branch) rb_list;
};
TAILQ_HEAD(rcs_dlist, rcs_delta);
struct rcs_delta {
RCSNUM *rd_num;
RCSNUM *rd_next;
int rd_flags;
struct tm rd_date;
char *rd_author;
char *rd_state;
char *rd_commitid;
char *rd_log;
char *rd_locker;
u_char *rd_text;
size_t rd_tlen;
TAILQ_HEAD(, rcs_branch) rd_branches;
TAILQ_ENTRY(rcs_delta) rd_list;
};
typedef struct rcs_file {
FILE *rf_file;
int rf_dead;
char *rf_path;
mode_t rf_mode;
int rf_flags;
RCSNUM *rf_head;
RCSNUM *rf_branch;
char *rf_comment;
char *rf_expand;
char *rf_desc;
u_int rf_ndelta;
struct rcs_dlist rf_delta;
TAILQ_HEAD(rcs_alist, rcs_access) rf_access;
TAILQ_HEAD(rcs_slist, rcs_sym) rf_symbols;
TAILQ_HEAD(rcs_llist, rcs_lock) rf_locks;
void *rf_pdata;
} RCSFILE;
struct rcs_line;
struct rcs_lines;
RCSFILE *rcs_open(const char *, int, int, ...);
void rcs_close(RCSFILE *);
RCSNUM *rcs_head_get(RCSFILE *);
int rcs_head_set(RCSFILE *, RCSNUM *);
RCSNUM *rcs_branch_new(RCSFILE *, RCSNUM *);
const RCSNUM *rcs_branch_get(RCSFILE *);
int rcs_branch_set(RCSFILE *, const RCSNUM *);
int rcs_access_add(RCSFILE *, const char *);
int rcs_access_remove(RCSFILE *, const char *);
struct rcs_delta *rcs_findrev(RCSFILE *, RCSNUM *);
int rcs_sym_add(RCSFILE *, const char *, RCSNUM *);
int rcs_sym_check(const char *);
struct rcs_sym *rcs_sym_get(RCSFILE *, const char *);
int rcs_sym_remove(RCSFILE *, const char *);
RCSNUM *rcs_sym_getrev(RCSFILE *, const char *);
RCSNUM *rcs_translate_tag(const char *, RCSFILE *);
int rcs_lock_getmode(RCSFILE *);
int rcs_lock_setmode(RCSFILE *, int);
int rcs_lock_add(RCSFILE *, const char *, RCSNUM *);
int rcs_lock_remove(RCSFILE *, const char *, RCSNUM *);
int rcs_deltatext_set(RCSFILE *, RCSNUM *, BUF *);
const char *rcs_desc_get(RCSFILE *);
void rcs_desc_set(RCSFILE *, const char *);
const char *rcs_comment_lookup(const char *);
const char *rcs_comment_get(RCSFILE *);
void rcs_comment_set(RCSFILE *, const char *);
void rcs_kwexp_set(RCSFILE *, int);
int rcs_kwexp_get(RCSFILE *);
int rcs_rev_add(RCSFILE *, RCSNUM *, const char *, time_t,
const char *);
time_t rcs_rev_getdate(RCSFILE *, RCSNUM *);
int rcs_rev_setlog(RCSFILE *, RCSNUM *, const char *);
int rcs_rev_remove(RCSFILE *, RCSNUM *);
int rcs_state_set(RCSFILE *, RCSNUM *, const char *);
const char *rcs_state_get(RCSFILE *, RCSNUM *);
int rcs_state_check(const char *);
void rcs_write(RCSFILE *);
int rcs_rev_write_stmp(RCSFILE *, RCSNUM *, char *, int);
void rcs_rev_write_fd(RCSFILE *, RCSNUM *, int, int);
struct rcs_lines *rcs_rev_getlines(RCSFILE *, RCSNUM *,
struct rcs_line ***);
void rcs_annotate_getlines(RCSFILE *, RCSNUM *,
struct rcs_line ***);
BUF *rcs_rev_getbuf(RCSFILE *, RCSNUM *, int);
void rcs_delta_stats(struct rcs_delta *, int *, int *);
int rcs_kflag_get(const char *);
RCSNUM *rcsnum_alloc(void);
RCSNUM *rcsnum_parse(const char *);
RCSNUM *rcsnum_brtorev(const RCSNUM *);
RCSNUM *rcsnum_revtobr(const RCSNUM *);
RCSNUM *rcsnum_inc(RCSNUM *);
RCSNUM *rcsnum_dec(RCSNUM *);
RCSNUM *rcsnum_branch_root(RCSNUM *);
RCSNUM *rcsnum_new_branch(RCSNUM *);
int rcsnum_addmagic(RCSNUM *);
int rcsnum_aton(const char *, const char **, RCSNUM *);
char *rcsnum_tostr(const RCSNUM *, char *, size_t);
void rcsnum_cpy(const RCSNUM *, RCSNUM *, u_int);
int rcsnum_cmp(RCSNUM *, RCSNUM *, u_int);
int rcsnum_differ(RCSNUM *, RCSNUM *);
#endif