aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--net/bridge/netfilter/ebtables.c33
1 files changed, 21 insertions, 12 deletions
diff --git a/net/bridge/netfilter/ebtables.c b/net/bridge/netfilter/ebtables.c
index f0d9ffd4c916..00a89705c1c4 100644
--- a/net/bridge/netfilter/ebtables.c
+++ b/net/bridge/netfilter/ebtables.c
@@ -1156,38 +1156,47 @@ int ebt_register_table(struct ebt_table *table)
1156{ 1156{
1157 struct ebt_table_info *newinfo; 1157 struct ebt_table_info *newinfo;
1158 struct ebt_table *t; 1158 struct ebt_table *t;
1159 struct ebt_replace *repl;
1159 int ret, i, countersize; 1160 int ret, i, countersize;
1161 void *p;
1160 1162
1161 if (!table || !table->table ||!table->table->entries || 1163 if (!table || !(repl = table->table) || !repl->entries ||
1162 table->table->entries_size == 0 || 1164 repl->entries_size == 0 ||
1163 table->table->counters || table->private) { 1165 repl->counters || table->private) {
1164 BUGPRINT("Bad table data for ebt_register_table!!!\n"); 1166 BUGPRINT("Bad table data for ebt_register_table!!!\n");
1165 return -EINVAL; 1167 return -EINVAL;
1166 } 1168 }
1167 1169
1168 countersize = COUNTER_OFFSET(table->table->nentries) * 1170 countersize = COUNTER_OFFSET(repl->nentries) *
1169 (highest_possible_processor_id()+1); 1171 (highest_possible_processor_id()+1);
1170 newinfo = vmalloc(sizeof(*newinfo) + countersize); 1172 newinfo = vmalloc(sizeof(*newinfo) + countersize);
1171 ret = -ENOMEM; 1173 ret = -ENOMEM;
1172 if (!newinfo) 1174 if (!newinfo)
1173 return -ENOMEM; 1175 return -ENOMEM;
1174 1176
1175 newinfo->entries = vmalloc(table->table->entries_size); 1177 p = vmalloc(repl->entries_size);
1176 if (!(newinfo->entries)) 1178 if (!p)
1177 goto free_newinfo; 1179 goto free_newinfo;
1178 1180
1179 memcpy(newinfo->entries, table->table->entries, 1181 memcpy(p, repl->entries, repl->entries_size);
1180 table->table->entries_size); 1182 newinfo->entries = p;
1183
1184 newinfo->entries_size = repl->entries_size;
1185 newinfo->nentries = repl->nentries;
1181 1186
1182 if (countersize) 1187 if (countersize)
1183 memset(newinfo->counters, 0, countersize); 1188 memset(newinfo->counters, 0, countersize);
1184 1189
1185 /* fill in newinfo and parse the entries */ 1190 /* fill in newinfo and parse the entries */
1186 newinfo->chainstack = NULL; 1191 newinfo->chainstack = NULL;
1187 ret = ebt_verify_pointers(table->table, newinfo); 1192 for (i = 0; i < NF_BR_NUMHOOKS; i++) {
1188 if (ret != 0) 1193 if ((repl->valid_hooks & (1 << i)) == 0)
1189 goto free_chainstack; 1194 newinfo->hook_entry[i] = NULL;
1190 ret = translate_table(table->table->name, newinfo); 1195 else
1196 newinfo->hook_entry[i] = p +
1197 ((char *)repl->hook_entry[i] - repl->entries);
1198 }
1199 ret = translate_table(repl->name, newinfo);
1191 if (ret != 0) { 1200 if (ret != 0) {
1192 BUGPRINT("Translate_table failed\n"); 1201 BUGPRINT("Translate_table failed\n");
1193 goto free_chainstack; 1202 goto free_chainstack;