141edb306SCy Schubert
241edb306SCy Schubert /*
341edb306SCy Schubert * Copyright (C) 2012 by Darren Reed.
441edb306SCy Schubert *
541edb306SCy Schubert * See the IPFILTER.LICENCE file for details on licencing.
641edb306SCy Schubert */
741edb306SCy Schubert #include <stdio.h>
841edb306SCy Schubert #include <unistd.h>
941edb306SCy Schubert #include <string.h>
1041edb306SCy Schubert #include <fcntl.h>
1141edb306SCy Schubert #include <errno.h>
1241edb306SCy Schubert #if !defined(__SVR4) && !defined(__GNUC__)
1341edb306SCy Schubert #include <strings.h>
1441edb306SCy Schubert #endif
1541edb306SCy Schubert #include <sys/types.h>
1641edb306SCy Schubert #include <sys/param.h>
1741edb306SCy Schubert #include <sys/file.h>
1841edb306SCy Schubert #include <stdlib.h>
1941edb306SCy Schubert #include <stddef.h>
2041edb306SCy Schubert #include <sys/socket.h>
2141edb306SCy Schubert #include <sys/ioctl.h>
2241edb306SCy Schubert #include <netinet/in.h>
2341edb306SCy Schubert #include <netinet/in_systm.h>
2441edb306SCy Schubert #include <sys/time.h>
2541edb306SCy Schubert #include <net/if.h>
2641edb306SCy Schubert #include <netinet/ip.h>
2741edb306SCy Schubert #include <netdb.h>
2841edb306SCy Schubert #include <arpa/nameser.h>
2941edb306SCy Schubert #include <resolv.h>
3041edb306SCy Schubert #include "ipf.h"
3141edb306SCy Schubert #include "netinet/ipl.h"
3241edb306SCy Schubert
3341edb306SCy Schubert
3441edb306SCy Schubert #ifndef IPF_SAVEDIR
3541edb306SCy Schubert # define IPF_SAVEDIR "/var/db/ipf"
3641edb306SCy Schubert #endif
3741edb306SCy Schubert #ifndef IPF_NATFILE
3841edb306SCy Schubert # define IPF_NATFILE "ipnat.ipf"
3941edb306SCy Schubert #endif
4041edb306SCy Schubert #ifndef IPF_STATEFILE
4141edb306SCy Schubert # define IPF_STATEFILE "ipstate.ipf"
4241edb306SCy Schubert #endif
4341edb306SCy Schubert
4441edb306SCy Schubert #if !defined(__SVR4) && defined(__GNUC__)
4541edb306SCy Schubert extern char *index(const char *, int);
4641edb306SCy Schubert #endif
4741edb306SCy Schubert
4841edb306SCy Schubert extern char *optarg;
4941edb306SCy Schubert extern int optind;
5041edb306SCy Schubert
5141edb306SCy Schubert int main(int, char *[]);
5241edb306SCy Schubert void usage(void);
5341edb306SCy Schubert int changestateif(char *, char *);
5441edb306SCy Schubert int changenatif(char *, char *);
5541edb306SCy Schubert int readstate(int, char *);
5641edb306SCy Schubert int readnat(int, char *);
5741edb306SCy Schubert int writestate(int, char *);
5841edb306SCy Schubert int opendevice(char *);
5941edb306SCy Schubert void closedevice(int);
6041edb306SCy Schubert int setlock(int, int);
6141edb306SCy Schubert int writeall(char *);
6241edb306SCy Schubert int readall(char *);
6341edb306SCy Schubert int writenat(int, char *);
6441edb306SCy Schubert
6541edb306SCy Schubert int opts = 0;
6641edb306SCy Schubert char *progname;
6741edb306SCy Schubert
6841edb306SCy Schubert
usage()6941edb306SCy Schubert void usage()
7041edb306SCy Schubert {
7141edb306SCy Schubert fprintf(stderr, "usage: %s [-nv] -l\n", progname);
7241edb306SCy Schubert fprintf(stderr, "usage: %s [-nv] -u\n", progname);
7341edb306SCy Schubert fprintf(stderr, "usage: %s [-nv] [-d <dir>] -R\n", progname);
7441edb306SCy Schubert fprintf(stderr, "usage: %s [-nv] [-d <dir>] -W\n", progname);
7541edb306SCy Schubert fprintf(stderr, "usage: %s [-nNSv] [-f <file>] -r\n", progname);
7641edb306SCy Schubert fprintf(stderr, "usage: %s [-nNSv] [-f <file>] -w\n", progname);
7741edb306SCy Schubert fprintf(stderr, "usage: %s [-nNSv] -f <filename> -i <if1>,<if2>\n",
7841edb306SCy Schubert progname);
7941edb306SCy Schubert exit(1);
8041edb306SCy Schubert }
8141edb306SCy Schubert
8241edb306SCy Schubert
8341edb306SCy Schubert /*
8441edb306SCy Schubert * Change interface names in state information saved out to disk.
8541edb306SCy Schubert */
changestateif(char * ifs,char * fname)86efeb8bffSCy Schubert int changestateif(char *ifs, char *fname)
8741edb306SCy Schubert {
8841edb306SCy Schubert int fd, olen, nlen, rw;
8941edb306SCy Schubert ipstate_save_t ips;
9041edb306SCy Schubert off_t pos;
9141edb306SCy Schubert char *s;
9241edb306SCy Schubert
9341edb306SCy Schubert s = strchr(ifs, ',');
9441edb306SCy Schubert if (!s)
9541edb306SCy Schubert usage();
9641edb306SCy Schubert *s++ = '\0';
9741edb306SCy Schubert nlen = strlen(s);
9841edb306SCy Schubert olen = strlen(ifs);
9941edb306SCy Schubert if (nlen >= sizeof(ips.ips_is.is_ifname) ||
10041edb306SCy Schubert olen >= sizeof(ips.ips_is.is_ifname))
10141edb306SCy Schubert usage();
10241edb306SCy Schubert
10341edb306SCy Schubert fd = open(fname, O_RDWR);
10441edb306SCy Schubert if (fd == -1) {
10541edb306SCy Schubert perror("open");
10641edb306SCy Schubert exit(1);
10741edb306SCy Schubert }
10841edb306SCy Schubert
10941edb306SCy Schubert for (pos = 0; read(fd, &ips, sizeof(ips)) == sizeof(ips); ) {
11041edb306SCy Schubert rw = 0;
11141edb306SCy Schubert if (!strncmp(ips.ips_is.is_ifname[0], ifs, olen + 1)) {
11241edb306SCy Schubert strcpy(ips.ips_is.is_ifname[0], s);
11341edb306SCy Schubert rw = 1;
11441edb306SCy Schubert }
11541edb306SCy Schubert if (!strncmp(ips.ips_is.is_ifname[1], ifs, olen + 1)) {
11641edb306SCy Schubert strcpy(ips.ips_is.is_ifname[1], s);
11741edb306SCy Schubert rw = 1;
11841edb306SCy Schubert }
11941edb306SCy Schubert if (!strncmp(ips.ips_is.is_ifname[2], ifs, olen + 1)) {
12041edb306SCy Schubert strcpy(ips.ips_is.is_ifname[2], s);
12141edb306SCy Schubert rw = 1;
12241edb306SCy Schubert }
12341edb306SCy Schubert if (!strncmp(ips.ips_is.is_ifname[3], ifs, olen + 1)) {
12441edb306SCy Schubert strcpy(ips.ips_is.is_ifname[3], s);
12541edb306SCy Schubert rw = 1;
12641edb306SCy Schubert }
12741edb306SCy Schubert if (rw == 1) {
12841edb306SCy Schubert if (lseek(fd, pos, SEEK_SET) != pos) {
12941edb306SCy Schubert perror("lseek");
13041edb306SCy Schubert exit(1);
13141edb306SCy Schubert }
13241edb306SCy Schubert if (write(fd, &ips, sizeof(ips)) != sizeof(ips)) {
13341edb306SCy Schubert perror("write");
13441edb306SCy Schubert exit(1);
13541edb306SCy Schubert }
13641edb306SCy Schubert }
13741edb306SCy Schubert pos = lseek(fd, 0, SEEK_CUR);
13841edb306SCy Schubert }
13941edb306SCy Schubert close(fd);
14041edb306SCy Schubert
141*2582ae57SCy Schubert return (0);
14241edb306SCy Schubert }
14341edb306SCy Schubert
14441edb306SCy Schubert
14541edb306SCy Schubert /*
14641edb306SCy Schubert * Change interface names in NAT information saved out to disk.
14741edb306SCy Schubert */
changenatif(char * ifs,char * fname)148efeb8bffSCy Schubert int changenatif(char *ifs, char *fname)
14941edb306SCy Schubert {
15041edb306SCy Schubert int fd, olen, nlen, rw;
15141edb306SCy Schubert nat_save_t ipn;
15241edb306SCy Schubert nat_t *nat;
15341edb306SCy Schubert off_t pos;
15441edb306SCy Schubert char *s;
15541edb306SCy Schubert
15641edb306SCy Schubert s = strchr(ifs, ',');
15741edb306SCy Schubert if (!s)
15841edb306SCy Schubert usage();
15941edb306SCy Schubert *s++ = '\0';
16041edb306SCy Schubert nlen = strlen(s);
16141edb306SCy Schubert olen = strlen(ifs);
16241edb306SCy Schubert nat = &ipn.ipn_nat;
16341edb306SCy Schubert if (nlen >= sizeof(nat->nat_ifnames[0]) ||
16441edb306SCy Schubert olen >= sizeof(nat->nat_ifnames[0]))
16541edb306SCy Schubert usage();
16641edb306SCy Schubert
16741edb306SCy Schubert fd = open(fname, O_RDWR);
16841edb306SCy Schubert if (fd == -1) {
16941edb306SCy Schubert perror("open");
17041edb306SCy Schubert exit(1);
17141edb306SCy Schubert }
17241edb306SCy Schubert
17341edb306SCy Schubert for (pos = 0; read(fd, &ipn, sizeof(ipn)) == sizeof(ipn); ) {
17441edb306SCy Schubert rw = 0;
17541edb306SCy Schubert if (!strncmp(nat->nat_ifnames[0], ifs, olen + 1)) {
17641edb306SCy Schubert strcpy(nat->nat_ifnames[0], s);
17741edb306SCy Schubert rw = 1;
17841edb306SCy Schubert }
17941edb306SCy Schubert if (!strncmp(nat->nat_ifnames[1], ifs, olen + 1)) {
18041edb306SCy Schubert strcpy(nat->nat_ifnames[1], s);
18141edb306SCy Schubert rw = 1;
18241edb306SCy Schubert }
18341edb306SCy Schubert if (rw == 1) {
18441edb306SCy Schubert if (lseek(fd, pos, SEEK_SET) != pos) {
18541edb306SCy Schubert perror("lseek");
18641edb306SCy Schubert exit(1);
18741edb306SCy Schubert }
18841edb306SCy Schubert if (write(fd, &ipn, sizeof(ipn)) != sizeof(ipn)) {
18941edb306SCy Schubert perror("write");
19041edb306SCy Schubert exit(1);
19141edb306SCy Schubert }
19241edb306SCy Schubert }
19341edb306SCy Schubert pos = lseek(fd, 0, SEEK_CUR);
19441edb306SCy Schubert }
19541edb306SCy Schubert close(fd);
19641edb306SCy Schubert
197*2582ae57SCy Schubert return (0);
19841edb306SCy Schubert }
19941edb306SCy Schubert
20041edb306SCy Schubert
main(int argc,char * argv[])201efeb8bffSCy Schubert int main(int argc, char *argv[])
20241edb306SCy Schubert {
20341edb306SCy Schubert int c, lock = -1, devfd = -1, err = 0, rw = -1, ns = -1, set = 0;
20441edb306SCy Schubert char *dirname = NULL, *filename = NULL, *ifs = NULL;
20541edb306SCy Schubert
20641edb306SCy Schubert progname = argv[0];
20741edb306SCy Schubert while ((c = getopt(argc, argv, "d:f:i:lNnSRruvWw")) != -1)
20841edb306SCy Schubert switch (c)
20941edb306SCy Schubert {
21041edb306SCy Schubert case 'd' :
21141edb306SCy Schubert if ((set == 0) && !dirname && !filename)
21241edb306SCy Schubert dirname = optarg;
21341edb306SCy Schubert else
21441edb306SCy Schubert usage();
21541edb306SCy Schubert break;
21641edb306SCy Schubert case 'f' :
21741edb306SCy Schubert if ((set != 0) && !dirname && !filename)
21841edb306SCy Schubert filename = optarg;
21941edb306SCy Schubert else
22041edb306SCy Schubert usage();
22141edb306SCy Schubert break;
22241edb306SCy Schubert case 'i' :
22341edb306SCy Schubert ifs = optarg;
22441edb306SCy Schubert set = 1;
22541edb306SCy Schubert break;
22641edb306SCy Schubert case 'l' :
22741edb306SCy Schubert if (filename || dirname || set)
22841edb306SCy Schubert usage();
22941edb306SCy Schubert lock = 1;
23041edb306SCy Schubert set = 1;
23141edb306SCy Schubert break;
23241edb306SCy Schubert case 'n' :
23341edb306SCy Schubert opts |= OPT_DONOTHING;
23441edb306SCy Schubert break;
23541edb306SCy Schubert case 'N' :
23641edb306SCy Schubert if ((ns >= 0) || dirname || (rw != -1) || set)
23741edb306SCy Schubert usage();
23841edb306SCy Schubert ns = 0;
23941edb306SCy Schubert set = 1;
24041edb306SCy Schubert break;
24141edb306SCy Schubert case 'r' :
24241edb306SCy Schubert if (dirname || (rw != -1) || (ns == -1))
24341edb306SCy Schubert usage();
24441edb306SCy Schubert rw = 0;
24541edb306SCy Schubert set = 1;
24641edb306SCy Schubert break;
24741edb306SCy Schubert case 'R' :
24841edb306SCy Schubert rw = 2;
24941edb306SCy Schubert set = 1;
25041edb306SCy Schubert break;
25141edb306SCy Schubert case 'S' :
25241edb306SCy Schubert if ((ns >= 0) || dirname || (rw != -1) || set)
25341edb306SCy Schubert usage();
25441edb306SCy Schubert ns = 1;
25541edb306SCy Schubert set = 1;
25641edb306SCy Schubert break;
25741edb306SCy Schubert case 'u' :
25841edb306SCy Schubert if (filename || dirname || set)
25941edb306SCy Schubert usage();
26041edb306SCy Schubert lock = 0;
26141edb306SCy Schubert set = 1;
26241edb306SCy Schubert break;
26341edb306SCy Schubert case 'v' :
26441edb306SCy Schubert opts |= OPT_VERBOSE;
26541edb306SCy Schubert break;
26641edb306SCy Schubert case 'w' :
26741edb306SCy Schubert if (dirname || (rw != -1) || (ns == -1))
26841edb306SCy Schubert usage();
26941edb306SCy Schubert rw = 1;
27041edb306SCy Schubert set = 1;
27141edb306SCy Schubert break;
27241edb306SCy Schubert case 'W' :
27341edb306SCy Schubert rw = 3;
27441edb306SCy Schubert set = 1;
27541edb306SCy Schubert break;
27641edb306SCy Schubert case '?' :
27741edb306SCy Schubert default :
27841edb306SCy Schubert usage();
27941edb306SCy Schubert }
28041edb306SCy Schubert
28141edb306SCy Schubert if (ifs) {
28241edb306SCy Schubert if (!filename || ns < 0)
28341edb306SCy Schubert usage();
28441edb306SCy Schubert if (ns == 0)
285*2582ae57SCy Schubert return (changenatif(ifs, filename));
28641edb306SCy Schubert else
287*2582ae57SCy Schubert return (changestateif(ifs, filename));
28841edb306SCy Schubert }
28941edb306SCy Schubert
29041edb306SCy Schubert if ((ns >= 0) || (lock >= 0)) {
29141edb306SCy Schubert if (lock >= 0)
29241edb306SCy Schubert devfd = opendevice(NULL);
29341edb306SCy Schubert else if (ns >= 0) {
29441edb306SCy Schubert if (ns == 1)
29541edb306SCy Schubert devfd = opendevice(IPSTATE_NAME);
29641edb306SCy Schubert else if (ns == 0)
29741edb306SCy Schubert devfd = opendevice(IPNAT_NAME);
29841edb306SCy Schubert }
29941edb306SCy Schubert if (devfd == -1)
30041edb306SCy Schubert exit(1);
30141edb306SCy Schubert }
30241edb306SCy Schubert
30341edb306SCy Schubert if (lock >= 0)
30441edb306SCy Schubert err = setlock(devfd, lock);
30541edb306SCy Schubert else if (rw >= 0) {
30641edb306SCy Schubert if (rw & 1) { /* WRITE */
30741edb306SCy Schubert if (rw & 2)
30841edb306SCy Schubert err = writeall(dirname);
30941edb306SCy Schubert else {
31041edb306SCy Schubert if (ns == 0)
31141edb306SCy Schubert err = writenat(devfd, filename);
31241edb306SCy Schubert else if (ns == 1)
31341edb306SCy Schubert err = writestate(devfd, filename);
31441edb306SCy Schubert }
31541edb306SCy Schubert } else {
31641edb306SCy Schubert if (rw & 2)
31741edb306SCy Schubert err = readall(dirname);
31841edb306SCy Schubert else {
31941edb306SCy Schubert if (ns == 0)
32041edb306SCy Schubert err = readnat(devfd, filename);
32141edb306SCy Schubert else if (ns == 1)
32241edb306SCy Schubert err = readstate(devfd, filename);
32341edb306SCy Schubert }
32441edb306SCy Schubert }
32541edb306SCy Schubert }
326*2582ae57SCy Schubert return (err);
32741edb306SCy Schubert }
32841edb306SCy Schubert
32941edb306SCy Schubert
opendevice(char * ipfdev)330efeb8bffSCy Schubert int opendevice(char *ipfdev)
33141edb306SCy Schubert {
33241edb306SCy Schubert int fd = -1;
33341edb306SCy Schubert
33441edb306SCy Schubert if (opts & OPT_DONOTHING)
335*2582ae57SCy Schubert return (-2);
33641edb306SCy Schubert
33741edb306SCy Schubert if (!ipfdev)
33841edb306SCy Schubert ipfdev = IPL_NAME;
33941edb306SCy Schubert
34041edb306SCy Schubert if ((fd = open(ipfdev, O_RDWR)) == -1)
34141edb306SCy Schubert if ((fd = open(ipfdev, O_RDONLY)) == -1)
34241edb306SCy Schubert perror("open device");
343*2582ae57SCy Schubert return (fd);
34441edb306SCy Schubert }
34541edb306SCy Schubert
34641edb306SCy Schubert
closedevice(int fd)347efeb8bffSCy Schubert void closedevice(int fd)
34841edb306SCy Schubert {
34941edb306SCy Schubert close(fd);
35041edb306SCy Schubert }
35141edb306SCy Schubert
35241edb306SCy Schubert
setlock(int fd,int lock)353efeb8bffSCy Schubert int setlock(int fd, int lock)
35441edb306SCy Schubert {
35541edb306SCy Schubert if (opts & OPT_VERBOSE)
35641edb306SCy Schubert printf("Turn lock %s\n", lock ? "on" : "off");
35741edb306SCy Schubert if (!(opts & OPT_DONOTHING)) {
35841edb306SCy Schubert if (ioctl(fd, SIOCSTLCK, &lock) == -1) {
35941edb306SCy Schubert perror("SIOCSTLCK");
360*2582ae57SCy Schubert return (1);
36141edb306SCy Schubert }
36241edb306SCy Schubert if (opts & OPT_VERBOSE)
36341edb306SCy Schubert printf("Lock now %s\n", lock ? "on" : "off");
36441edb306SCy Schubert }
365*2582ae57SCy Schubert return (0);
36641edb306SCy Schubert }
36741edb306SCy Schubert
36841edb306SCy Schubert
writestate(int fd,char * file)369efeb8bffSCy Schubert int writestate(int fd, char *file)
37041edb306SCy Schubert {
37141edb306SCy Schubert ipstate_save_t ips, *ipsp;
37241edb306SCy Schubert ipfobj_t obj;
37341edb306SCy Schubert int wfd = -1;
37441edb306SCy Schubert
37541edb306SCy Schubert if (!file)
37641edb306SCy Schubert file = IPF_STATEFILE;
37741edb306SCy Schubert
37841edb306SCy Schubert wfd = open(file, O_WRONLY|O_TRUNC|O_CREAT, 0600);
37941edb306SCy Schubert if (wfd == -1) {
38041edb306SCy Schubert fprintf(stderr, "%s ", file);
38141edb306SCy Schubert perror("state:open");
382*2582ae57SCy Schubert return (1);
38341edb306SCy Schubert }
38441edb306SCy Schubert
38541edb306SCy Schubert ipsp = &ips;
38641edb306SCy Schubert bzero((char *)&obj, sizeof(obj));
38741edb306SCy Schubert bzero((char *)ipsp, sizeof(ips));
38841edb306SCy Schubert
38941edb306SCy Schubert obj.ipfo_rev = IPFILTER_VERSION;
39041edb306SCy Schubert obj.ipfo_size = sizeof(*ipsp);
39141edb306SCy Schubert obj.ipfo_type = IPFOBJ_STATESAVE;
39241edb306SCy Schubert obj.ipfo_ptr = ipsp;
39341edb306SCy Schubert
39441edb306SCy Schubert do {
39541edb306SCy Schubert
39641edb306SCy Schubert if (opts & OPT_VERBOSE)
39741edb306SCy Schubert printf("Getting state from addr %p\n", ips.ips_next);
39841edb306SCy Schubert if (ioctl(fd, SIOCSTGET, &obj)) {
39941edb306SCy Schubert if (errno == ENOENT)
40041edb306SCy Schubert break;
40141edb306SCy Schubert perror("state:SIOCSTGET");
40241edb306SCy Schubert close(wfd);
403*2582ae57SCy Schubert return (1);
40441edb306SCy Schubert }
40541edb306SCy Schubert if (opts & OPT_VERBOSE)
40641edb306SCy Schubert printf("Got state next %p\n", ips.ips_next);
40741edb306SCy Schubert if (write(wfd, ipsp, sizeof(ips)) != sizeof(ips)) {
40841edb306SCy Schubert perror("state:write");
40941edb306SCy Schubert close(wfd);
410*2582ae57SCy Schubert return (1);
41141edb306SCy Schubert }
41241edb306SCy Schubert } while (ips.ips_next != NULL);
41341edb306SCy Schubert close(wfd);
41441edb306SCy Schubert
415*2582ae57SCy Schubert return (0);
41641edb306SCy Schubert }
41741edb306SCy Schubert
41841edb306SCy Schubert
readstate(int fd,char * file)419efeb8bffSCy Schubert int readstate(int fd, char *file)
42041edb306SCy Schubert {
42141edb306SCy Schubert ipstate_save_t ips, *is, *ipshead = NULL, *is1, *ipstail = NULL;
42241edb306SCy Schubert int sfd = -1, i;
42341edb306SCy Schubert ipfobj_t obj;
42441edb306SCy Schubert
42541edb306SCy Schubert if (!file)
42641edb306SCy Schubert file = IPF_STATEFILE;
42741edb306SCy Schubert
42841edb306SCy Schubert sfd = open(file, O_RDONLY, 0600);
42941edb306SCy Schubert if (sfd == -1) {
43041edb306SCy Schubert fprintf(stderr, "%s ", file);
43141edb306SCy Schubert perror("open");
432*2582ae57SCy Schubert return (1);
43341edb306SCy Schubert }
43441edb306SCy Schubert
43541edb306SCy Schubert bzero((char *)&ips, sizeof(ips));
43641edb306SCy Schubert
43741edb306SCy Schubert /*
43841edb306SCy Schubert * 1. Read all state information in.
43941edb306SCy Schubert */
44041edb306SCy Schubert do {
44141edb306SCy Schubert i = read(sfd, &ips, sizeof(ips));
44241edb306SCy Schubert if (i == -1) {
44341edb306SCy Schubert perror("read");
44441edb306SCy Schubert goto freeipshead;
44541edb306SCy Schubert }
44641edb306SCy Schubert if (i == 0)
44741edb306SCy Schubert break;
44841edb306SCy Schubert if (i != sizeof(ips)) {
44941edb306SCy Schubert fprintf(stderr, "state:incomplete read: %d != %d\n",
45041edb306SCy Schubert i, (int)sizeof(ips));
45141edb306SCy Schubert goto freeipshead;
45241edb306SCy Schubert }
45341edb306SCy Schubert is = (ipstate_save_t *)malloc(sizeof(*is));
45441edb306SCy Schubert if (is == NULL) {
45541edb306SCy Schubert fprintf(stderr, "malloc failed\n");
45641edb306SCy Schubert goto freeipshead;
45741edb306SCy Schubert }
45841edb306SCy Schubert
45941edb306SCy Schubert bcopy((char *)&ips, (char *)is, sizeof(ips));
46041edb306SCy Schubert
46141edb306SCy Schubert /*
46241edb306SCy Schubert * Check to see if this is the first state entry that will
46341edb306SCy Schubert * reference a particular rule and if so, flag it as such
46441edb306SCy Schubert * else just adjust the rule pointer to become a pointer to
46541edb306SCy Schubert * the other. We do this so we have a means later for tracking
46641edb306SCy Schubert * who is referencing us when we get back the real pointer
46741edb306SCy Schubert * in is_rule after doing the ioctl.
46841edb306SCy Schubert */
46941edb306SCy Schubert for (is1 = ipshead; is1 != NULL; is1 = is1->ips_next)
47041edb306SCy Schubert if (is1->ips_rule == is->ips_rule)
47141edb306SCy Schubert break;
47241edb306SCy Schubert if (is1 == NULL)
47341edb306SCy Schubert is->ips_is.is_flags |= SI_NEWFR;
47441edb306SCy Schubert else
47541edb306SCy Schubert is->ips_rule = (void *)&is1->ips_rule;
47641edb306SCy Schubert
47741edb306SCy Schubert /*
47841edb306SCy Schubert * Use a tail-queue type list (add things to the end)..
47941edb306SCy Schubert */
48041edb306SCy Schubert is->ips_next = NULL;
48141edb306SCy Schubert if (!ipshead)
48241edb306SCy Schubert ipshead = is;
48341edb306SCy Schubert if (ipstail)
48441edb306SCy Schubert ipstail->ips_next = is;
48541edb306SCy Schubert ipstail = is;
48641edb306SCy Schubert } while (1);
48741edb306SCy Schubert
48841edb306SCy Schubert close(sfd);
48941edb306SCy Schubert
49041edb306SCy Schubert obj.ipfo_rev = IPFILTER_VERSION;
49141edb306SCy Schubert obj.ipfo_size = sizeof(*is);
49241edb306SCy Schubert obj.ipfo_type = IPFOBJ_STATESAVE;
49341edb306SCy Schubert
49441edb306SCy Schubert while ((is = ipshead) != NULL) {
49541edb306SCy Schubert if (opts & OPT_VERBOSE)
49641edb306SCy Schubert printf("Loading new state table entry\n");
49741edb306SCy Schubert if (is->ips_is.is_flags & SI_NEWFR) {
49841edb306SCy Schubert if (opts & OPT_VERBOSE)
49941edb306SCy Schubert printf("Loading new filter rule\n");
50041edb306SCy Schubert }
50141edb306SCy Schubert
50241edb306SCy Schubert obj.ipfo_ptr = is;
50341edb306SCy Schubert if (!(opts & OPT_DONOTHING))
50441edb306SCy Schubert if (ioctl(fd, SIOCSTPUT, &obj)) {
50541edb306SCy Schubert perror("SIOCSTPUT");
50641edb306SCy Schubert goto freeipshead;
50741edb306SCy Schubert }
50841edb306SCy Schubert
50941edb306SCy Schubert if (is->ips_is.is_flags & SI_NEWFR) {
51041edb306SCy Schubert if (opts & OPT_VERBOSE)
51141edb306SCy Schubert printf("Real rule addr %p\n", is->ips_rule);
51241edb306SCy Schubert for (is1 = is->ips_next; is1; is1 = is1->ips_next)
51341edb306SCy Schubert if (is1->ips_rule == (frentry_t *)&is->ips_rule)
51441edb306SCy Schubert is1->ips_rule = is->ips_rule;
51541edb306SCy Schubert }
51641edb306SCy Schubert
51741edb306SCy Schubert ipshead = is->ips_next;
51841edb306SCy Schubert free(is);
51941edb306SCy Schubert }
52041edb306SCy Schubert
521*2582ae57SCy Schubert return (0);
52241edb306SCy Schubert
52341edb306SCy Schubert freeipshead:
52441edb306SCy Schubert while ((is = ipshead) != NULL) {
52541edb306SCy Schubert ipshead = is->ips_next;
52641edb306SCy Schubert free(is);
52741edb306SCy Schubert }
52841edb306SCy Schubert if (sfd != -1)
52941edb306SCy Schubert close(sfd);
530*2582ae57SCy Schubert return (1);
53141edb306SCy Schubert }
53241edb306SCy Schubert
53341edb306SCy Schubert
readnat(int fd,char * file)534efeb8bffSCy Schubert int readnat(int fd, char *file)
53541edb306SCy Schubert {
53641edb306SCy Schubert nat_save_t ipn, *in, *ipnhead = NULL, *in1, *ipntail = NULL;
53741edb306SCy Schubert ipfobj_t obj;
53841edb306SCy Schubert int nfd, i;
53941edb306SCy Schubert nat_t *nat;
54041edb306SCy Schubert char *s;
54141edb306SCy Schubert int n;
54241edb306SCy Schubert
54341edb306SCy Schubert nfd = -1;
54441edb306SCy Schubert in = NULL;
54541edb306SCy Schubert ipnhead = NULL;
54641edb306SCy Schubert ipntail = NULL;
54741edb306SCy Schubert
54841edb306SCy Schubert if (!file)
54941edb306SCy Schubert file = IPF_NATFILE;
55041edb306SCy Schubert
55141edb306SCy Schubert nfd = open(file, O_RDONLY);
55241edb306SCy Schubert if (nfd == -1) {
55341edb306SCy Schubert fprintf(stderr, "%s ", file);
55441edb306SCy Schubert perror("nat:open");
555*2582ae57SCy Schubert return (1);
55641edb306SCy Schubert }
55741edb306SCy Schubert
55841edb306SCy Schubert bzero((char *)&ipn, sizeof(ipn));
55941edb306SCy Schubert
56041edb306SCy Schubert /*
56141edb306SCy Schubert * 1. Read all state information in.
56241edb306SCy Schubert */
56341edb306SCy Schubert do {
56441edb306SCy Schubert i = read(nfd, &ipn, sizeof(ipn));
56541edb306SCy Schubert if (i == -1) {
56641edb306SCy Schubert perror("read");
56741edb306SCy Schubert goto freenathead;
56841edb306SCy Schubert }
56941edb306SCy Schubert if (i == 0)
57041edb306SCy Schubert break;
57141edb306SCy Schubert if (i != sizeof(ipn)) {
57241edb306SCy Schubert fprintf(stderr, "nat:incomplete read: %d != %d\n",
57341edb306SCy Schubert i, (int)sizeof(ipn));
57441edb306SCy Schubert goto freenathead;
57541edb306SCy Schubert }
57641edb306SCy Schubert
57741edb306SCy Schubert in = (nat_save_t *)malloc(ipn.ipn_dsize);
57841edb306SCy Schubert if (in == NULL) {
57941edb306SCy Schubert fprintf(stderr, "nat:cannot malloc nat save atruct\n");
58041edb306SCy Schubert goto freenathead;
58141edb306SCy Schubert }
58241edb306SCy Schubert
58341edb306SCy Schubert if (ipn.ipn_dsize > sizeof(ipn)) {
58441edb306SCy Schubert n = ipn.ipn_dsize - sizeof(ipn);
58541edb306SCy Schubert if (n > 0) {
58641edb306SCy Schubert s = in->ipn_data + sizeof(in->ipn_data);
58741edb306SCy Schubert i = read(nfd, s, n);
58841edb306SCy Schubert if (i == 0)
58941edb306SCy Schubert break;
59041edb306SCy Schubert if (i != n) {
59141edb306SCy Schubert fprintf(stderr,
59241edb306SCy Schubert "nat:incomplete read: %d != %d\n",
59341edb306SCy Schubert i, n);
59441edb306SCy Schubert goto freenathead;
59541edb306SCy Schubert }
59641edb306SCy Schubert }
59741edb306SCy Schubert }
59841edb306SCy Schubert bcopy((char *)&ipn, (char *)in, sizeof(ipn));
59941edb306SCy Schubert
60041edb306SCy Schubert /*
60141edb306SCy Schubert * Check to see if this is the first NAT entry that will
60241edb306SCy Schubert * reference a particular rule and if so, flag it as such
60341edb306SCy Schubert * else just adjust the rule pointer to become a pointer to
60441edb306SCy Schubert * the other. We do this so we have a means later for tracking
60541edb306SCy Schubert * who is referencing us when we get back the real pointer
60641edb306SCy Schubert * in is_rule after doing the ioctl.
60741edb306SCy Schubert */
60841edb306SCy Schubert nat = &in->ipn_nat;
60941edb306SCy Schubert if (nat->nat_fr != NULL) {
61041edb306SCy Schubert for (in1 = ipnhead; in1 != NULL; in1 = in1->ipn_next)
61141edb306SCy Schubert if (in1->ipn_rule == nat->nat_fr)
61241edb306SCy Schubert break;
61341edb306SCy Schubert if (in1 == NULL)
61441edb306SCy Schubert nat->nat_flags |= SI_NEWFR;
61541edb306SCy Schubert else
61641edb306SCy Schubert nat->nat_fr = &in1->ipn_fr;
61741edb306SCy Schubert }
61841edb306SCy Schubert
61941edb306SCy Schubert /*
62041edb306SCy Schubert * Use a tail-queue type list (add things to the end)..
62141edb306SCy Schubert */
62241edb306SCy Schubert in->ipn_next = NULL;
62341edb306SCy Schubert if (!ipnhead)
62441edb306SCy Schubert ipnhead = in;
62541edb306SCy Schubert if (ipntail)
62641edb306SCy Schubert ipntail->ipn_next = in;
62741edb306SCy Schubert ipntail = in;
62841edb306SCy Schubert } while (1);
62941edb306SCy Schubert
63041edb306SCy Schubert close(nfd);
63141edb306SCy Schubert nfd = -1;
63241edb306SCy Schubert
63341edb306SCy Schubert obj.ipfo_rev = IPFILTER_VERSION;
63441edb306SCy Schubert obj.ipfo_type = IPFOBJ_NATSAVE;
63541edb306SCy Schubert
63641edb306SCy Schubert while ((in = ipnhead) != NULL) {
63741edb306SCy Schubert if (opts & OPT_VERBOSE)
63841edb306SCy Schubert printf("Loading new NAT table entry\n");
63941edb306SCy Schubert nat = &in->ipn_nat;
64041edb306SCy Schubert if (nat->nat_flags & SI_NEWFR) {
64141edb306SCy Schubert if (opts & OPT_VERBOSE)
64241edb306SCy Schubert printf("Loading new filter rule\n");
64341edb306SCy Schubert }
64441edb306SCy Schubert
64541edb306SCy Schubert obj.ipfo_ptr = in;
64641edb306SCy Schubert obj.ipfo_size = in->ipn_dsize;
64741edb306SCy Schubert if (!(opts & OPT_DONOTHING))
64841edb306SCy Schubert if (ioctl(fd, SIOCSTPUT, &obj)) {
64941edb306SCy Schubert fprintf(stderr, "in=%p:", in);
65041edb306SCy Schubert perror("SIOCSTPUT");
651*2582ae57SCy Schubert return (1);
65241edb306SCy Schubert }
65341edb306SCy Schubert
65441edb306SCy Schubert if (nat->nat_flags & SI_NEWFR) {
65541edb306SCy Schubert if (opts & OPT_VERBOSE)
65641edb306SCy Schubert printf("Real rule addr %p\n", nat->nat_fr);
65741edb306SCy Schubert for (in1 = in->ipn_next; in1; in1 = in1->ipn_next)
65841edb306SCy Schubert if (in1->ipn_rule == &in->ipn_fr)
65941edb306SCy Schubert in1->ipn_rule = nat->nat_fr;
66041edb306SCy Schubert }
66141edb306SCy Schubert
66241edb306SCy Schubert ipnhead = in->ipn_next;
66341edb306SCy Schubert free(in);
66441edb306SCy Schubert }
66541edb306SCy Schubert
666*2582ae57SCy Schubert return (0);
66741edb306SCy Schubert
66841edb306SCy Schubert freenathead:
66941edb306SCy Schubert while ((in = ipnhead) != NULL) {
67041edb306SCy Schubert ipnhead = in->ipn_next;
67141edb306SCy Schubert free(in);
67241edb306SCy Schubert }
67341edb306SCy Schubert if (nfd != -1)
67441edb306SCy Schubert close(nfd);
675*2582ae57SCy Schubert return (1);
67641edb306SCy Schubert }
67741edb306SCy Schubert
67841edb306SCy Schubert
writenat(int fd,char * file)679efeb8bffSCy Schubert int writenat(int fd, char *file)
68041edb306SCy Schubert {
68141edb306SCy Schubert nat_save_t *ipnp = NULL, *next = NULL;
68241edb306SCy Schubert ipfobj_t obj;
68341edb306SCy Schubert int nfd = -1;
68441edb306SCy Schubert natget_t ng;
68541edb306SCy Schubert
68641edb306SCy Schubert if (!file)
68741edb306SCy Schubert file = IPF_NATFILE;
68841edb306SCy Schubert
68941edb306SCy Schubert nfd = open(file, O_WRONLY|O_TRUNC|O_CREAT, 0600);
69041edb306SCy Schubert if (nfd == -1) {
69141edb306SCy Schubert fprintf(stderr, "%s ", file);
69241edb306SCy Schubert perror("nat:open");
693*2582ae57SCy Schubert return (1);
69441edb306SCy Schubert }
69541edb306SCy Schubert
69641edb306SCy Schubert obj.ipfo_rev = IPFILTER_VERSION;
69741edb306SCy Schubert obj.ipfo_type = IPFOBJ_NATSAVE;
69841edb306SCy Schubert
69941edb306SCy Schubert do {
70041edb306SCy Schubert if (opts & OPT_VERBOSE)
70141edb306SCy Schubert printf("Getting nat from addr %p\n", ipnp);
70241edb306SCy Schubert ng.ng_ptr = next;
70341edb306SCy Schubert ng.ng_sz = 0;
70441edb306SCy Schubert if (ioctl(fd, SIOCSTGSZ, &ng)) {
70541edb306SCy Schubert perror("nat:SIOCSTGSZ");
70641edb306SCy Schubert close(nfd);
70741edb306SCy Schubert if (ipnp != NULL)
70841edb306SCy Schubert free(ipnp);
709*2582ae57SCy Schubert return (1);
71041edb306SCy Schubert }
71141edb306SCy Schubert
71241edb306SCy Schubert if (opts & OPT_VERBOSE)
71341edb306SCy Schubert printf("NAT size %d from %p\n", ng.ng_sz, ng.ng_ptr);
71441edb306SCy Schubert
71541edb306SCy Schubert if (ng.ng_sz == 0)
71641edb306SCy Schubert break;
71741edb306SCy Schubert
71841edb306SCy Schubert if (!ipnp)
71941edb306SCy Schubert ipnp = malloc(ng.ng_sz);
72041edb306SCy Schubert else
72141edb306SCy Schubert ipnp = realloc((char *)ipnp, ng.ng_sz);
72241edb306SCy Schubert if (!ipnp) {
72341edb306SCy Schubert fprintf(stderr,
72441edb306SCy Schubert "malloc for %d bytes failed\n", ng.ng_sz);
72541edb306SCy Schubert break;
72641edb306SCy Schubert }
72741edb306SCy Schubert
72841edb306SCy Schubert bzero((char *)ipnp, ng.ng_sz);
72941edb306SCy Schubert obj.ipfo_size = ng.ng_sz;
73041edb306SCy Schubert obj.ipfo_ptr = ipnp;
73141edb306SCy Schubert ipnp->ipn_dsize = ng.ng_sz;
73241edb306SCy Schubert ipnp->ipn_next = next;
73341edb306SCy Schubert if (ioctl(fd, SIOCSTGET, &obj)) {
73441edb306SCy Schubert if (errno == ENOENT)
73541edb306SCy Schubert break;
73641edb306SCy Schubert perror("nat:SIOCSTGET");
73741edb306SCy Schubert close(nfd);
73841edb306SCy Schubert free(ipnp);
739*2582ae57SCy Schubert return (1);
74041edb306SCy Schubert }
74141edb306SCy Schubert
74241edb306SCy Schubert if (opts & OPT_VERBOSE)
74341edb306SCy Schubert printf("Got nat next %p ipn_dsize %d ng_sz %d\n",
74441edb306SCy Schubert ipnp->ipn_next, ipnp->ipn_dsize, ng.ng_sz);
74541edb306SCy Schubert if (write(nfd, ipnp, ipnp->ipn_dsize) != ipnp->ipn_dsize) {
74641edb306SCy Schubert perror("nat:write");
74741edb306SCy Schubert close(nfd);
74841edb306SCy Schubert free(ipnp);
749*2582ae57SCy Schubert return (1);
75041edb306SCy Schubert }
75141edb306SCy Schubert next = ipnp->ipn_next;
75241edb306SCy Schubert } while (ipnp && next);
75341edb306SCy Schubert if (ipnp != NULL)
75441edb306SCy Schubert free(ipnp);
75541edb306SCy Schubert close(nfd);
75641edb306SCy Schubert
757*2582ae57SCy Schubert return (0);
75841edb306SCy Schubert }
75941edb306SCy Schubert
76041edb306SCy Schubert
writeall(char * dirname)761efeb8bffSCy Schubert int writeall(char *dirname)
76241edb306SCy Schubert {
76341edb306SCy Schubert int fd, devfd;
76441edb306SCy Schubert
76541edb306SCy Schubert if (!dirname)
76641edb306SCy Schubert dirname = IPF_SAVEDIR;
76741edb306SCy Schubert
76841edb306SCy Schubert if (chdir(dirname)) {
76941edb306SCy Schubert fprintf(stderr, "IPF_SAVEDIR=%s: ", dirname);
77041edb306SCy Schubert perror("chdir(IPF_SAVEDIR)");
771*2582ae57SCy Schubert return (1);
77241edb306SCy Schubert }
77341edb306SCy Schubert
77441edb306SCy Schubert fd = opendevice(NULL);
77541edb306SCy Schubert if (fd == -1)
776*2582ae57SCy Schubert return (1);
77741edb306SCy Schubert if (setlock(fd, 1)) {
77841edb306SCy Schubert close(fd);
779*2582ae57SCy Schubert return (1);
78041edb306SCy Schubert }
78141edb306SCy Schubert
78241edb306SCy Schubert devfd = opendevice(IPSTATE_NAME);
78341edb306SCy Schubert if (devfd == -1)
78441edb306SCy Schubert goto bad;
78541edb306SCy Schubert if (writestate(devfd, NULL))
78641edb306SCy Schubert goto bad;
78741edb306SCy Schubert close(devfd);
78841edb306SCy Schubert
78941edb306SCy Schubert devfd = opendevice(IPNAT_NAME);
79041edb306SCy Schubert if (devfd == -1)
79141edb306SCy Schubert goto bad;
79241edb306SCy Schubert if (writenat(devfd, NULL))
79341edb306SCy Schubert goto bad;
79441edb306SCy Schubert close(devfd);
79541edb306SCy Schubert
79641edb306SCy Schubert if (setlock(fd, 0)) {
79741edb306SCy Schubert close(fd);
798*2582ae57SCy Schubert return (1);
79941edb306SCy Schubert }
80041edb306SCy Schubert
80141edb306SCy Schubert close(fd);
802*2582ae57SCy Schubert return (0);
80341edb306SCy Schubert
80441edb306SCy Schubert bad:
80541edb306SCy Schubert setlock(fd, 0);
80641edb306SCy Schubert close(fd);
807*2582ae57SCy Schubert return (1);
80841edb306SCy Schubert }
80941edb306SCy Schubert
81041edb306SCy Schubert
readall(char * dirname)811efeb8bffSCy Schubert int readall(char *dirname)
81241edb306SCy Schubert {
81341edb306SCy Schubert int fd, devfd;
81441edb306SCy Schubert
81541edb306SCy Schubert if (!dirname)
81641edb306SCy Schubert dirname = IPF_SAVEDIR;
81741edb306SCy Schubert
81841edb306SCy Schubert if (chdir(dirname)) {
81941edb306SCy Schubert perror("chdir(IPF_SAVEDIR)");
820*2582ae57SCy Schubert return (1);
82141edb306SCy Schubert }
82241edb306SCy Schubert
82341edb306SCy Schubert fd = opendevice(NULL);
82441edb306SCy Schubert if (fd == -1)
825*2582ae57SCy Schubert return (1);
82641edb306SCy Schubert if (setlock(fd, 1)) {
82741edb306SCy Schubert close(fd);
828*2582ae57SCy Schubert return (1);
82941edb306SCy Schubert }
83041edb306SCy Schubert
83141edb306SCy Schubert devfd = opendevice(IPSTATE_NAME);
83241edb306SCy Schubert if (devfd == -1)
833*2582ae57SCy Schubert return (1);
83441edb306SCy Schubert if (readstate(devfd, NULL))
835*2582ae57SCy Schubert return (1);
83641edb306SCy Schubert close(devfd);
83741edb306SCy Schubert
83841edb306SCy Schubert devfd = opendevice(IPNAT_NAME);
83941edb306SCy Schubert if (devfd == -1)
840*2582ae57SCy Schubert return (1);
84141edb306SCy Schubert if (readnat(devfd, NULL))
842*2582ae57SCy Schubert return (1);
84341edb306SCy Schubert close(devfd);
84441edb306SCy Schubert
84541edb306SCy Schubert if (setlock(fd, 0)) {
84641edb306SCy Schubert close(fd);
847*2582ae57SCy Schubert return (1);
84841edb306SCy Schubert }
84941edb306SCy Schubert
850*2582ae57SCy Schubert return (0);
85141edb306SCy Schubert }
852