#ifndef RCS_H
#define RCS_H
#include <sys/queue.h>
#include <stdio.h>
#include "buf.h"
#define RCS_DIFF_MAXARG 32
#define RCS_DIFF_DIV \
"==================================================================="
#define RCSDIR "RCS"
#define RCS_FILE_EXT ",v"
#define RCS_HEAD_BRANCH "HEAD"
#define RCS_HEAD_INIT "1.1"
#define RCS_HEAD_REV ((RCSNUM *)(-1))
#define RCS_STATE_INVALCHAR "$,:;@"
#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 0x10000
#define RCS_KW_ID \
(RCS_KW_RCSFILE | RCS_KW_REVISION | RCS_KW_DATE \
| RCS_KW_AUTHOR | RCS_KW_STATE | RCS_KW_LOCKER)
#define RCS_KW_HEADER (RCS_KW_ID | RCS_KW_FULLPATH)
#define RCS_KWEXP_NONE 0x00
#define RCS_KWEXP_NAME 0x01
#define RCS_KWEXP_VAL 0x02
#define RCS_KWEXP_LKR 0x04
#define RCS_KWEXP_OLD 0x08
#define RCS_KWEXP_ERR 0x10
#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 64
#define RCSNUM_ISBRANCH(n) ((n)->rn_len % 2)
#define RCSNUM_ISBRANCHREV(n) (!((n)->rn_len % 2) && ((n)->rn_len >= 4))
#define RCSNUM_NO_MAGIC (1<<0)
#define RCS_READ (1<<0)
#define RCS_WRITE (1<<1)
#define RCS_RDWR (RCS_READ|RCS_WRITE)
#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_ERR_NOERR 0
#define RCS_ERR_NOENT 1
#define RCS_ERR_DUPENT 2
#define RCS_ERR_BADNUM 3
#define RCS_ERR_BADSYM 4
#define RCS_ERR_PARSE 5
#define RCS_ERR_ERRNO 255
#define CHECKOUT_REV_CREATED 1
#define CHECKOUT_REV_MERGED 2
#define CHECKOUT_REV_REMOVED 3
#define CHECKOUT_REV_UPDATED 4
#define RCS_COMMITID_MAXLEN 64
typedef struct rcs_num {
u_int rn_len;
u_int16_t *rn_id;
} 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;
u_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;
char *rf_path;
mode_t rf_mode;
u_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;
extern int rcs_errno;
RCSFILE *rcs_open(const char *, int, int, ...);
void rcs_close(RCSFILE *);
int rcs_head_set(RCSFILE *, RCSNUM *);
const RCSNUM *rcs_branch_get(RCSFILE *);
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_remove(RCSFILE *, const char *);
RCSNUM *rcs_sym_getrev(RCSFILE *, const char *);
int rcs_sym_check(const char *);
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 *);
BUF *rcs_getrev(RCSFILE *, RCSNUM *);
int rcs_deltatext_set(RCSFILE *, RCSNUM *, BUF *);
void rcs_desc_set(RCSFILE *, const char *);
void rcs_comment_set(RCSFILE *, const char *);
BUF *rcs_kwexp_buf(BUF *, RCSFILE *, RCSNUM *);
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 *);
int rcs_state_check(const char *);
void rcs_write(RCSFILE *);
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 *);
void rcsnum_free(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(const RCSNUM *, const RCSNUM *, u_int);
void rcs_set_tz(char *, struct rcs_delta *, struct tm *);
extern char *timezone_flag;
extern int rcsnum_flags;
#endif