1 /* Code to save the iptables state, in human readable-form. */ 2 /* (C) 1999 by Paul 'Rusty' Russell <rusty (at) rustcorp.com.au> and 3 * (C) 2000-2002 by Harald Welte <laforge (at) gnumonks.org> 4 * 5 * This code is distributed under the terms of GNU GPL v2 6 * 7 */ 8 #include <getopt.h> 9 #include <sys/errno.h> 10 #include <stdio.h> 11 #include <fcntl.h> 12 #include <stdlib.h> 13 #include <string.h> 14 #include <time.h> 15 #include <netdb.h> 16 #include "libiptc/libiptc.h" 17 #include "iptables.h" 18 #include "iptables-multi.h" 19 20 #ifndef NO_SHARED_LIBS 21 #include <dlfcn.h> 22 #endif 23 24 static int show_binary = 0, show_counters = 0; 25 26 static const struct option options[] = { 27 {.name = "binary", .has_arg = false, .val = 'b'}, 28 {.name = "counters", .has_arg = false, .val = 'c'}, 29 {.name = "dump", .has_arg = false, .val = 'd'}, 30 {.name = "table", .has_arg = true, .val = 't'}, 31 {.name = "modprobe", .has_arg = true, .val = 'M'}, 32 {NULL}, 33 }; 34 35 /* Debugging prototype. */ 36 static int for_each_table(int (*func)(const char *tablename)) 37 { 38 int ret = 1; 39 FILE *procfile = NULL; 40 char tablename[IPT_TABLE_MAXNAMELEN+1]; 41 42 procfile = fopen("/proc/net/ip_tables_names", "re"); 43 if (!procfile) 44 return ret; 45 46 while (fgets(tablename, sizeof(tablename), procfile)) { 47 if (tablename[strlen(tablename) - 1] != '\n') 48 xtables_error(OTHER_PROBLEM, 49 "Badly formed tablename `%s'\n", 50 tablename); 51 tablename[strlen(tablename) - 1] = '\0'; 52 ret &= func(tablename); 53 } 54 55 fclose(procfile); 56 return ret; 57 } 58 59 60 static int do_output(const char *tablename) 61 { 62 struct iptc_handle *h; 63 const char *chain = NULL; 64 65 if (!tablename) 66 return for_each_table(&do_output); 67 68 h = iptc_init(tablename); 69 if (h == NULL) { 70 xtables_load_ko(xtables_modprobe_program, false); 71 h = iptc_init(tablename); 72 } 73 if (!h) 74 xtables_error(OTHER_PROBLEM, "Cannot initialize: %s\n", 75 iptc_strerror(errno)); 76 77 if (!show_binary) { 78 time_t now = time(NULL); 79 80 printf("# Generated by iptables-save v%s on %s", 81 IPTABLES_VERSION, ctime(&now)); 82 printf("*%s\n", tablename); 83 84 /* Dump out chain names first, 85 * thereby preventing dependency conflicts */ 86 for (chain = iptc_first_chain(h); 87 chain; 88 chain = iptc_next_chain(h)) { 89 90 printf(":%s ", chain); 91 if (iptc_builtin(chain, h)) { 92 struct ipt_counters count; 93 printf("%s ", 94 iptc_get_policy(chain, &count, h)); 95 printf("[%llu:%llu]\n", (unsigned long long)count.pcnt, (unsigned long long)count.bcnt); 96 } else { 97 printf("- [0:0]\n"); 98 } 99 } 100 101 102 for (chain = iptc_first_chain(h); 103 chain; 104 chain = iptc_next_chain(h)) { 105 const struct ipt_entry *e; 106 107 /* Dump out rules */ 108 e = iptc_first_rule(chain, h); 109 while(e) { 110 print_rule4(e, h, chain, show_counters); 111 e = iptc_next_rule(e, h); 112 } 113 } 114 115 now = time(NULL); 116 printf("COMMIT\n"); 117 printf("# Completed on %s", ctime(&now)); 118 } else { 119 /* Binary, huh? OK. */ 120 xtables_error(OTHER_PROBLEM, "Binary NYI\n"); 121 } 122 123 iptc_free(h); 124 125 return 1; 126 } 127 128 /* Format: 129 * :Chain name POLICY packets bytes 130 * rule 131 */ 132 #ifdef IPTABLES_MULTI 133 int 134 iptables_save_main(int argc, char *argv[]) 135 #else 136 int 137 main(int argc, char *argv[]) 138 #endif 139 { 140 const char *tablename = NULL; 141 int c; 142 143 iptables_globals.program_name = "iptables-save"; 144 c = xtables_init_all(&iptables_globals, NFPROTO_IPV4); 145 if (c < 0) { 146 fprintf(stderr, "%s/%s Failed to initialize xtables\n", 147 iptables_globals.program_name, 148 iptables_globals.program_version); 149 exit(1); 150 } 151 #if defined(ALL_INCLUSIVE) || defined(NO_SHARED_LIBS) 152 init_extensions(); 153 init_extensions4(); 154 #endif 155 156 while ((c = getopt_long(argc, argv, "bcdt:", options, NULL)) != -1) { 157 switch (c) { 158 case 'b': 159 show_binary = 1; 160 break; 161 162 case 'c': 163 show_counters = 1; 164 break; 165 166 case 't': 167 /* Select specific table. */ 168 tablename = optarg; 169 break; 170 case 'M': 171 xtables_modprobe_program = optarg; 172 break; 173 case 'd': 174 do_output(tablename); 175 exit(0); 176 } 177 } 178 179 if (optind < argc) { 180 fprintf(stderr, "Unknown arguments found on commandline\n"); 181 exit(1); 182 } 183 184 return !do_output(tablename); 185 } 186