VLAN: Correct ID for RT-N66 allowing VLAN and tagged network.
[tomato.git] / release / src / router / www / advanced-vlan.asp
blobe982a59252d8e1d173dd726616b191ccf42cc03c
1 <!DOCTYPE HTML PUBLIC '-//W3C//DTD HTML 4.0//EN'>
2 <!--
3 Tomato VLAN GUI
4 Copyright (C) 2011-2012 Augusto Bott
5 http://code.google.com/p/tomato-sdhc-vlan/
7 Tomato GUI
8 Copyright (C) 2006-2007 Jonathan Zarate
9 http://www.polarcloud.com/tomato/
11 Tomato VLAN update and bug correction
12 Copyright (C) 2011-2012 Vicente Soriano
13 http://tomatoraf.com
15 For use with Tomato Firmware only.
16 No part of this file may be used without permission.
17 -->
18 <html>
19 <head>
20 <meta http-equiv='content-type' content='text/html;charset=utf-8'>
21 <meta name='robots' content='noindex,nofollow'>
22 <title>[<% ident(); %>] Advanced: VLAN</title>
23 <link rel='stylesheet' type='text/css' href='tomato.css'>
24 <% css(); %>
25 <script type='text/javascript' src='tomato.js'></script>
26 <style type='text/css'>
27 #vlan-grid .co1,
28 #vlan-grid .co2,
29 #vlan-grid .co3,
30 #vlan-grid .co4,
31 #vlan-grid .co5,
32 #vlan-grid .co6,
33 #vlan-grid .co7,
34 #vlan-grid .co8,
35 #vlan-grid .co9,
36 #vlan-grid .co10,
37 #vlan-grid .co11,
38 #vlan-grid .co12,
39 #vlan-grid .co13,
40 #vlan-grid .co14 {
41 text-align: center;
43 #vlan-grid .co2 {
44 width: 60px;
46 #vlan-grid .centered {
47 text-align: center;
49 </style>
50 <script type='text/javascript' src='wireless.jsx?_http_id=<% nv(http_id); %>'></script>
51 <script type='text/javascript' src='interfaces.js'></script>
52 <script type='text/javascript'>
53 <% nvram ("vlan0ports,vlan1ports,vlan2ports,vlan3ports,vlan4ports,vlan5ports,vlan6ports,vlan7ports,vlan8ports,vlan9ports,vlan10ports,vlan11ports,vlan12ports,vlan13ports,vlan14ports,vlan15ports,vlan0hwname,vlan1hwname,vlan2hwname,vlan3hwname,vlan4hwname,vlan5hwname,vlan6hwname,vlan7hwname,vlan8hwname,vlan9hwname,vlan10hwname,vlan11hwname,vlan12hwname,vlan13hwname,vlan14hwname,vlan15hwname,wan_ifnameX,manual_boot_nv,boardtype,boardflags,trunk_vlan_so,lan_ifname,lan_ifnames,lan1_ifname,lan1_ifnames,lan2_ifname,lan2_ifnames,lan3_ifname,lan3_ifnames,boardrev,vlan0tag,vlan0vid,vlan1vid,vlan2vid,vlan3vid,vlan4vid,vlan5vid,vlan6vid,vlan7vid,vlan8vid,vlan9vid,vlan10vid,vlan11vid,vlan12vid,vlan13vid,vlan14vid,vlan15vid");%>
55 var port_vlan_supported = 0;
56 var trunk_vlan_supported = 0;
58 // does not seem to be strictly necessary for boardflags as it's supposed to be a bitmap
59 nvram['boardflags'] = ((nvram['boardflags'].toLowerCase().indexOf('0x') != -1) ? '0x' : '') + String('0000' + ((nvram['boardflags'].toLowerCase()).replace('0x',''))).slice(-4);
60 // but the contents of router/shared/id.c seem to indicate string formatting/padding might be required for some models as we check if strings match
61 nvram['boardtype'] = ((nvram['boardtype'].toLowerCase().indexOf('0x') != -1) ? '0x' : '') + String('0000' + ((nvram['boardtype'].toLowerCase()).replace('0x',''))).slice(-4);
63 // see http://www.dd-wrt.com/wiki/index.php/Hardware#Boardflags and router/shared/id.c
64 if(nvram['boardflags'] & 0x0100) { // BFL_ENETVLAN = this board has vlan capability
65 port_vlan_supported = 1;
68 // TESTED ONLY ON WRT54G v2 (boardtype 0x0101) and WRT54GL v1.1 (boardtype 0x0467)
69 // attempt of cross-referencing boardtypes/routers mentioned on id.c and the wiki page above
70 switch(nvram['boardtype']) {
71 case '0x0467': // WRT54GL 1.x, WRT54GS 3.x/4.x
72 case '0x048e': // WL-520GU, WL-500G Premium v2
73 case '0x04ef': // WRT320N/E2000
74 case '0x04cf': // WRT610Nv2/E3000, RT-N16
75 case '0xf52c': // E4200v1
76 case '0xf5b2': // RT-N66
77 trunk_vlan_supported = 1;
78 break;
79 default:
80 break;
83 // TESTED ONLY ON WRT54G v2 (boardtype 0x0101),WRT54GL v1.1 (boardtype 0x0467) and WNR3500L (boardtype 0x04cf)
84 // info on some of these boardtypes/routers obtained from
85 // http://wiki.openwrt.org/toh/asus/start
86 // http://wiki.openwrt.org/toh/linksys/start
87 // http://wiki.openwrt.org/toh/start
88 switch(nvram['boardtype']) {
89 case '0x0467': // WRT54GL 1.x, WRT54GS 3.x/4.x
90 if (nvram['boardrev'] == '0x13') { // WHR-G54S
91 COL_P0N = '1';
92 COL_P1N = '2';
93 COL_P2N = '3';
94 COL_P3N = '4';
95 COL_P4N = '0';
96 break;
98 case '0xa4cf': // Belkin F7D3301
99 if (nvram['boardrev'] == '0x1100'){ //Belkin F5D8235-4 v3
100 COL_P0N = '1';
101 COL_P1N = '2';
102 COL_P2N = '3';
103 COL_P3N = '4';
104 COL_P4N = '0';
105 break;
107 case '0xd4cf': // Belkin F7D4301
108 case '0x048e': // WL-520GU, WL-500G Premium v2
109 case '0x0550': // RT-N53 (boardrev = 0x1442), RT-N10U ( boardrev = 0x1102)
110 if (nvram['boardrev'] == '0x1102') { //RT-N10U
111 COL_P0N = '1';
112 COL_P1N = '2';
113 COL_P2N = '3';
114 COL_P3N = '4';
115 COL_P4N = '0';
116 break;
118 if (nvram['boardrev'] == '0x1100') { //CW-5358U
119 COL_P0N = '1';
120 COL_P1N = '2';
121 COL_P2N = '3';
122 COL_P3N = '4';
123 COL_P4N = '0';
124 break;
126 COL_P0N = '3';
127 COL_P1N = '2';
128 COL_P2N = '1';
129 COL_P3N = '0';
130 COL_P4N = '4';
131 break;
132 case '0x04ef': // WRT320N/E2000
133 case '0x04cf': // WRT610Nv2/E3000, RT-N16, WNR3500L
134 case '0xf5b2': // RT-N66
135 COL_P0N = '4';
136 COL_P1N = '3';
137 COL_P2N = '2';
138 COL_P3N = '1';
139 COL_P4N = '0';
140 break;
141 case '0xf53a': // E1000v2.1/E1200v1
142 case '0xf53b': // E1000v2/E1500
143 if (((nvram['boot_hw_model'] == 'E1200') && (nvram['boot_hw_ver'] == '1.0')) || (nvram['boot_hw_model'] == 'E1500')) {
144 COL_P0N = '0';
145 COL_P1N = '1';
146 COL_P2N = '2';
147 COL_P3N = '3';
148 COL_P4N = '4';
149 break;
151 COL_P0N = '1';
152 COL_P1N = '2';
153 COL_P2N = '3';
154 COL_P3N = '4';
155 COL_P4N = '0';
156 break;
157 case '0xc550': // E1550
158 case '0xf550': // E2500
159 case '0x058e': // E900
160 case '0xf52a': // E3200
161 case '0xf52c': // E4200v1
162 case '0x1202': // HG320 - not sure, need test
163 if (nvram['boardrev'] == '0x1153') { //RG200E-CA type 0x058e same as E900
164 COL_P0N = '4';
165 COL_P1N = '3';
166 COL_P2N = '2';
167 COL_P3N = '1';
168 COL_P4N = '0';
169 break;
171 COL_P0N = '0';
172 COL_P1N = '1';
173 COL_P2N = '2';
174 COL_P3N = '3';
175 COL_P4N = '4';
176 break;
177 case '0x052b':
178 if (nvram['boardrev'] == '02') { //WNR3500Lv2
179 COL_P0N = '4';
180 COL_P1N = '3';
181 COL_P2N = '2';
182 COL_P3N = '1';
183 COL_P4N = '0';
184 break;
186 if (nvram['boardrev'] == '0x1204') { //rt-n15u
187 COL_P0N = '3';
188 COL_P1N = '2';
189 COL_P2N = '1';
190 COL_P3N = '0';
191 COL_P4N = '4';
192 break;
194 // should work on WRT54G v2/v3, WRT54GS v1/v2 and others
195 default:
196 COL_P0N = '1';
197 COL_P1N = '2';
198 COL_P2N = '3';
199 COL_P3N = '4';
200 COL_P4N = '0';
201 break;
204 var COL_VID = 0;
205 var COL_MAP = 1;
206 var COL_P0 = 2;
207 var COL_P0T = 3;
208 var COL_P1 = 4;
209 var COL_P1T = 5;
210 var COL_P2 = 6;
211 var COL_P2T = 7;
212 var COL_P3 = 8;
213 var COL_P3T = 9;
214 var COL_P4 = 10;
215 var COL_P4T = 11;
216 var COL_VID_DEF = 12;
217 var COL_BRI = 13;
219 var vlt = nvram.vlan0tag | '0';
221 // set to either 5 or 8 when nvram settings are read (FastE or GigE routers)
222 var SWITCH_INTERNAL_PORT=0;
223 // option made available for experimental purposes on routers known to support port-based VLANs, but not confirmed to support 801.11q trunks
224 var PORT_VLAN_SUPPORT_OVERRIDE = ((nvram['trunk_vlan_so'] == '1') ? 1 : 0);
226 function verifyFields(focused, quiet){
227 PORT_VLAN_SUPPORT_OVERRIDE=(E('_f_trunk_vlan_so').checked ? 1 : 0);
228 for (var uidx = 0; uidx < wl_ifaces.length; ++uidx) {
229 var u = wl_fface(uidx);
230 var wlan = E('_f_bridge_wlan'+u+'_to');
231 if(nvram.lan_ifname.length < 1)
232 wlan.options[0].disabled=true;
233 if(nvram.lan1_ifname.length < 1)
234 wlan.options[1].disabled=true;
235 if(nvram.lan2_ifname.length < 1)
236 wlan.options[2].disabled=true;
237 if(nvram.lan3_ifname.length < 1)
238 wlan.options[3].disabled=true;
240 var e = E('_vlan0tag');
241 if (!v_range('_vlan0tag', quiet, 0, 4080)) return 0;
242 var v = parseInt(e.value);
243 e.value = v - (v % 16);
244 if ((e.value != vlt) && (typeof(vlg) != 'undefined')) {
245 vlg.populate();
246 vlt = e.value;
248 return 1;
251 function save() {
252 if (vlg.isEditing()) return;
254 var fom = E('_fom');
255 fom.trunk_vlan_so.value = (E('_f_trunk_vlan_so').checked ? 1 : 0);
256 // wipe out relevant fields just in case this is not the first time we try to submit
257 for (var i = 0 ; i <= MAX_VLAN_ID ; i++) {
258 fom['vlan' + i + 'ports'].value = '';
259 fom['vlan' + i + 'hwname'].value = '';
260 fom['vlan' + i + 'vid'].value = '';
262 fom['wan_ifnameX'].value = '';
263 fom['lan_ifnames'].value = '';
264 fom['lan1_ifnames'].value = '';
265 fom['lan2_ifnames'].value = '';
266 fom['lan3_ifnames'].value = '';
268 var v = '';
269 var d = vlg.getAllData();
271 for (var i = 0; i < d.length; ++i) {
272 var p = '';
273 p += (d[i][COL_P0].toString() != '0') ? COL_P0N : '';
274 p += (((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (d[i][COL_P0T].toString() != '0')) ? 't' : '';
275 p += trailingSpace(p);
277 p += (d[i][COL_P1].toString() != '0') ? COL_P1N : '';
278 p += (((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (d[i][COL_P1T].toString() != '0')) ? 't' : '';
279 p += trailingSpace(p);
281 p += (d[i][COL_P2].toString() != '0') ? COL_P2N : '';
282 p += (((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (d[i][COL_P2T].toString() != '0')) ? 't' : '';
283 p += trailingSpace(p);
285 p += (d[i][COL_P3].toString() != '0') ? COL_P3N : '';
286 p += (((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (d[i][COL_P3T].toString() != '0')) ? 't' : '';
287 p += trailingSpace(p);
289 p += (d[i][COL_P4].toString() != '0') ? COL_P4N : '';
290 p += (((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (d[i][COL_P4T].toString() != '0')) ? 't' : '';
291 p += trailingSpace(p);
293 p += (d[i][COL_VID_DEF].toString() != '0') ? (SWITCH_INTERNAL_PORT + '*') : SWITCH_INTERNAL_PORT;
295 // arrange port numbers in ascending order just to be safe (not sure if this is really needed... mostly, cosmetics?)
296 p = p.split(" ");
297 p = p.sort(cmpInt);
298 p = p.join(" ");
300 v += (d[i][COL_VID_DEF].toString() != '0') ? d[i][0] : '';
302 fom['vlan'+d[i][COL_VID]+'ports'].value = p;
303 fom['vlan'+d[i][COL_VID]+'hwname'].value = 'et0';
304 fom['vlan'+d[i][COL_VID]+'vid'].value = ((d[i][COL_MAP].toString() != '') && (d[i][COL_MAP].toString() != '0')) ? d[i][COL_MAP] : '';
306 fom['wan_ifnameX'].value += (d[i][COL_BRI] == '2') ? 'vlan'+d[i][0] : '';
307 fom['lan_ifnames'].value += (d[i][COL_BRI] == '3') ? 'vlan'+d[i][0] : '';
308 /* REMOVE-BEGIN
309 // fom['lan_ifnames'].value += trailingSpace(fom['lan_ifnames'].value);
310 // alert('vlan'+d[i][0]+'ports='+fom['vlan'+d[i][0]+'ports'].value+'\nvlan'+d[i][0]+'hwname='+fom['vlan'+d[i][0]+'hwname'].value);
311 REMOVE-END */
312 fom['lan1_ifnames'].value += (d[i][COL_BRI] == '4') ? 'vlan'+d[i][0] : '';
313 fom['lan2_ifnames'].value += (d[i][COL_BRI] == '5') ? 'vlan'+d[i][0] : '';
314 fom['lan3_ifnames'].value += (d[i][COL_BRI] == '6') ? 'vlan'+d[i][0] : '';
317 for (var uidx = 0; uidx < wl_ifaces.length; ++uidx) {
318 var u = wl_fface(uidx);
319 var wlan = E('_f_bridge_wlan'+u+'_to');
320 /* REMOVE-BEGIN
321 // var wlan = E('_f_bridge_wlan_to');
322 // alert(wlan.selectedIndex);
323 REMOVE-END */
324 switch(parseInt(wlan.selectedIndex)) {
325 case 0:
326 fom['lan_ifnames'].value += ' ' + wl_ifaces[uidx][0];
327 break;
328 case 1:
329 fom['lan1_ifnames'].value += ' ' + wl_ifaces[uidx][0];
330 break;
331 case 2:
332 fom['lan2_ifnames'].value += ' ' + wl_ifaces[uidx][0];
333 break;
334 case 3:
335 fom['lan3_ifnames'].value += ' ' + wl_ifaces[uidx][0];
336 break;
339 /* REMOVE-BEGIN
340 // var lif = nvram['lan_ifnames'].split(' ');
341 // for (var j = 0; j < lif.length; j++) {
342 // fom['lan_ifnames'].value += (lif[j].indexOf('vlan') != -1) ? '' : lif[j];
343 // fom['lan_ifnames'].value += trailingSpace(fom['lan_ifnames'].value);
344 // }
345 // alert('lan_ifnames=' + fom['lan_ifnames'].value + '\n' +
346 // 'lan1_ifnames=' + fom['lan1_ifnames'].value + '\n' +
347 // 'lan2_ifnames=' + fom['lan2_ifnames'].value + '\n' +
348 // 'lan3_ifnames=' + fom['lan3_ifnames'].value);
349 REMOVE-END */
351 // for some models, Tomato checks for a few vital/crucial nvram settings at init time
352 // in some cases, if some/any of them are not found, a full nvram reset/clean could be triggered
353 // so, to (try to) play it safe, we check for the 1st needed/available/required
354 // VLAN for FastE (vlan0 is usually LAN) and GigE routers (vlan1 is usually LAN)
355 if((fom['vlan0ports'].value.length < 1) || (fom['vlan0hwname'].value.length < 1) ||
356 (fom['vlan1ports'].value.length < 1) || (fom['vlan1hwname'].value.length < 1))
357 fom['manual_boot_nv'].value = '1';
358 else
359 fom['manual_boot_nv'].value = nvram['manual_boot_nv'];
361 var e = E('footer-msg');
363 if(vlg.countWan() != 1) {
364 e.innerHTML = 'Cannot proceed: one VID must be assigned to WAN.';
365 e.style.visibility = 'visible';
366 setTimeout(
367 function() {
368 e.innerHTML = '';
369 e.style.visibility = 'hidden';
370 }, 5000);
371 return;
374 if(vlg.countLan(0) != 1) {
375 e.innerHTML = 'Cannot proceed: one and only one VID must be assigned to the primary LAN (br0).';
376 e.style.visibility = 'visible';
377 setTimeout(
378 function() {
379 e.innerHTML = '';
380 e.style.visibility = 'hidden';
381 }, 5000);
382 return;
385 if (v.length < 1) {
386 e.innerHTML = 'Cannot proceed without setting a default VID';
387 e.style.visibility = 'visible';
388 setTimeout(
389 function() {
390 e.innerHTML = '';
391 e.style.visibility = 'hidden';
392 }, 5000);
393 return;
396 if (confirm("Router must be rebooted to proceed. Commit changes to NVRAM and reboot now?"))
397 form.submit(fom, 0);
400 function trailingSpace(s)
402 return ((s.length>0)&&(s.charAt(s.length-1) != ' ')) ? ' ' : '';
405 if(port_vlan_supported) { // aka if(supported_hardware) block
406 var vlg = new TomatoGrid();
407 vlg.setup = function() {
408 this.init('vlan-grid', '', (MAX_VLAN_ID + 1), [
409 { type: 'select', options: [[0, '0'],[1, '1'],[2, '2'],[3, '3'],[4, '4'],[5, '5'],[6, '6'],[7, '7'],[8, '8'],[9, '9'],[10, '10'],[11, '11'],[12, '12'],[13, '13'],[14, '14'],[15, '15']], prefix: '<div class="centered">', suffix: '</div>' },
410 { type: 'text', maxlen: 4, prefix: '<div class="centered">', suffix: '</div>' },
411 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
412 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
413 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
414 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
415 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
416 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
417 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
418 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
419 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
420 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
421 { type: 'checkbox', prefix: '<div class="centered">', suffix: '</div>' },
422 { type: 'select', options: [[1, 'none'],[2, 'WAN'],[3, 'LAN (br0)'],[4, 'LAN1 (br1)'],[5, 'LAN2 (br2)'],[6, 'LAN3 (br3)']], prefix: '<div class="centered">', suffix: '</div>' }]);
424 this.headerSet(['VLAN', 'VID', 'Port 1', 'Tagged', 'Port 2', 'Tagged', 'Port 3', 'Tagged', 'Port 4', 'Tagged', 'WAN Port', 'Tagged', 'Default', 'Bridge']);
426 vlg.populate();
427 vlg.canDelete = false;
428 vlg.sort(0);
429 vlg.showNewEditor();
430 vlg.resetNewEditor();
433 vlg.populate = function() {
434 vlg.removeAllData();
436 // find out which vlans are supposed to be bridged to each LAN
437 var bridged = [];
439 for (var i = 0 ; i <= MAX_BRIDGE_ID ; i++) {
440 var j = (i == 0) ? '' : i.toString();
441 var l = nvram['lan' + j + '_ifnames'].split(' ');
442 /* REMOVE-BEGIN
443 // alert('lan' + j + '_ifnames=' + l);
444 REMOVE-END */
445 for (var k = 0 ; k < l.length; k++) {
446 /* REMOVE-BEGIN
447 // alert("bridge br" + i + "=vlan" + parseInt(l[k].replace('vlan','')));
448 REMOVE-END */
449 if(l[k].indexOf('vlan') != -1) {
450 /* REMOVE-BEGIN
451 // alert('lan' + j + '_ifname=br' + nvram['lan' + j + '_ifname'].replace('br',''));
452 REMOVE-END */
453 if (nvram['lan' + j + '_ifname'] != '')
454 bridged[parseInt(l[k].replace('vlan',''))] = (3 + parseInt(nvram['lan' + j + '_ifname'].replace('br',''))).toString();
455 else
456 bridged[parseInt(l[k].replace('vlan',''))] = '1';
458 // WLAN
459 for (var uidx = 0; uidx < wl_ifaces.length; ++uidx) {
460 if(l[k].indexOf(wl_ifaces[uidx][0]) != -1) {
461 E('_f_bridge_wlan'+wl_fface(uidx)+'_to').selectedIndex=i;
467 // WAN port
468 bridged[parseInt(nvram['wan_ifnameX'].replace('vlan',''))] = '2';
470 // go thru all possible VLANs
471 for (var i = 0 ; i <= MAX_VLAN_ID ; i++) {
472 var port = [];
473 var tagged = [];
474 if ((nvram['vlan' + i + 'hwname'].length > 0) || (nvram['vlan' + i + 'ports'].length > 0)) {
475 // (re)initialize our bitmap for this particular iteration
476 for (var j=0; j <= MAX_PORT_ID ; j++) {
477 port[j] = '0';
478 tagged[j] = '0';
480 // which ports are members of this VLAN?
481 var m=nvram['vlan' + i + 'ports'].split(' ');
482 for (var j = 0; j < (m.length) ; j++) {
483 port[parseInt(m[j].charAt(0))] = '1';
484 tagged[parseInt(m[j].charAt(0))] = (((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (m[j].indexOf('t') != -1)) ? '1' : '0';
487 if (port_vlan_supported) {
488 if((nvram['vlan' + i + 'ports']).indexOf('*') != -1)
489 SWITCH_INTERNAL_PORT=(nvram['vlan' + i + 'ports']).charAt((nvram['vlan' + i + 'ports']).indexOf('*')-1);
491 vlg.insertData(-1, [ i.toString(),
492 ((nvram['vlan' + i + 'vid'] != '') && (nvram['vlan' + i + 'vid'] > 0)) ? (nvram['vlan' + i + 'vid']).toString() : '0',
493 port[COL_P0N], tagged[COL_P0N],
494 port[COL_P1N], tagged[COL_P1N],
495 port[COL_P2N], tagged[COL_P2N],
496 port[COL_P3N], tagged[COL_P3N],
497 port[COL_P4N], tagged[COL_P4N],
498 (((nvram['vlan' + i + 'ports']).indexOf('*') != -1) ? '1' : '0' ),
499 (bridged[i] != null) ? bridged[i] : '1' ]);
505 vlg.countElem = function(f, v)
507 var data = this.getAllData();
508 var total = 0;
509 for (var i = 0; i < data.length; ++i) {
510 total += (data[i][f] == v) ? 1 : 0;
512 return total;
515 vlg.countDefaultVID = function()
517 return this.countElem(COL_VID_DEF,1);
520 vlg.countVID = function (v)
522 return this.countElem(COL_VID,v);
525 vlg.countWan = function()
527 return this.countElem(COL_BRI,2);
530 vlg.countLan = function(l)
532 return this.countElem(COL_BRI,l+3);
535 vlg.verifyFields = function(row, quiet) {
536 var valid = 1;
537 var f = fields.getAll(row);
539 for(var i=0; i<= MAX_VLAN_ID ; i++) {
540 f[COL_VID].options[i].disabled = (this.countVID(i) > 0);
543 for (var i=0; i <= MAX_BRIDGE_ID; i++) {
544 var j = (i==0) ? '' : i.toString();
545 f[COL_BRI].options[i+2].disabled = (nvram['lan' + j + '_ifname'].length < 1);
548 if (!v_range(f[COL_MAP], quiet, 0, 4094)) valid = 0;
550 if(((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (f[COL_P0].checked == 1)) {
551 f[COL_P0T].disabled=0;
552 /* REMOVE-BEGIN
553 // if((f[COL_P0T].checked==0) || (this.countElem(COL_P0,1)>0) )
554 // if(this.countElem(COL_P0,1)>0) {
555 // }
556 REMOVE-END */
557 } else {
558 f[COL_P0T].disabled=1;
559 f[COL_P0T].checked=0;
561 if(((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (f[COL_P1].checked == 1)) {
562 f[COL_P1T].disabled=0;
563 } else {
564 f[COL_P1T].disabled=1;
565 f[COL_P1T].checked=0;
567 if(((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (f[COL_P2].checked == 1)) {
568 f[COL_P2T].disabled=0;
569 } else {
570 f[COL_P2T].disabled=1;
571 f[COL_P2T].checked=0;
573 if(((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (f[COL_P3].checked == 1)) {
574 f[COL_P3T].disabled=0;
575 } else {
576 f[COL_P3T].disabled=1;
577 f[COL_P3T].checked=0;
579 if(((trunk_vlan_supported) || (PORT_VLAN_SUPPORT_OVERRIDE)) && (f[COL_P4].checked == 1)) {
580 f[COL_P4T].disabled=0;
581 } else {
582 f[COL_P4T].disabled=1;
583 f[COL_P4T].checked=0;
586 if ((f[COL_P0].checked == 1) && (this.countElem(COL_P0,1)>0)) {
587 if (((this.countElem(COL_P0,1) != this.countElem(COL_P0T,1)) || (f[COL_P0T].checked==0))) {
588 ferror.set(f[COL_P0T], 'Port 1 cannot be assigned to more than one VLAN unless frames are tagged on all VLANs Port 1 is member', quiet);
589 valid=0;
590 } else {
591 ferror.clear(f[COL_P0T]);
594 if ((f[COL_P1].checked == 1) && (this.countElem(COL_P1,1)>0)) {
595 if (((this.countElem(COL_P1,1) != this.countElem(COL_P1T,1)) || (f[COL_P1T].checked==0))) {
596 ferror.set(f[COL_P1T], 'Port 2 cannot be assigned to more than one VLAN unless frames are tagged on all VLANs Port 2 is member', quiet);
597 valid=0;
598 } else {
599 ferror.clear(f[COL_P1T]);
602 if ((f[COL_P2].checked == 1) && (this.countElem(COL_P2,1)>0)) {
603 if (((this.countElem(COL_P2,1) != this.countElem(COL_P2T,1)) || (f[COL_P2T].checked==0))) {
604 ferror.set(f[COL_P2T], 'Port 3 cannot be assigned to more than one VLAN unless frames are tagged on all VLANs Port 3 is member', quiet);
605 valid=0;
606 } else {
607 ferror.clear(f[COL_P2T]);
610 if ((f[COL_P3].checked == 1) && (this.countElem(COL_P3,1)>0)) {
611 if (((this.countElem(COL_P3,1) != this.countElem(COL_P3T,1)) || (f[COL_P3T].checked==0))) {
612 ferror.set(f[COL_P3T], 'Port 4 cannot be assigned to more than one VLAN unless frames are tagged on all VLANs Port 4 is member', quiet);
613 valid=0;
614 } else {
615 ferror.clear(f[COL_P3T]);
618 if ((f[COL_P4].checked == 1) && (this.countElem(COL_P4,1)>0)) {
619 if (((this.countElem(COL_P4,1) != this.countElem(COL_P4T,1)) || (f[COL_P4T].checked==0))) {
620 ferror.set(f[COL_P4T], 'WAN port cannot be assigned to more than one VLAN unless frames are tagged on all VLANs WAN port is member', quiet);
621 valid=0;
622 } else {
623 ferror.clear(f[COL_P4T]);
627 if(this.countDefaultVID() > 0) {
628 f[COL_VID_DEF].disabled=1;
629 f[COL_VID_DEF].checked=0;
632 if((this.countDefaultVID() > 0) && (f[COL_VID_DEF].checked ==1)) {
633 ferror.set(f[COL_VID_DEF], 'Only one VID can be selected as the default VID', quiet);
634 valid = 0;
635 } else {
636 ferror.clear(f[COL_VID_DEF]);
639 if(this.countVID(f[COL_VID].selectedIndex) > 0) {
640 ferror.set(f[COL_VID], 'Cannot add more than one entry with VID ' + f[0].selectedIndex, quiet);
641 valid = 0;
642 } else {
643 ferror.clear(f[COL_VID]);
646 if ((this.countWan() > 0) && (f[COL_BRI].selectedIndex == 1)) {
647 ferror.set(f[COL_BRI],'Only one VID can be used as WAN at any time', quiet);
648 valid = 0;
649 } else {
650 ferror.clear(f[COL_BRI]);
653 for(var i=0; i<4; i++) {
654 if ((this.countLan(i) > 0) && (f[COL_BRI].selectedIndex == (i+2))) {
655 ferror.set(f[COL_BRI],'One and only one VID can be used for LAN' + ((i==0) ? '' : i ) + ' (br'+i+') at any time', quiet);
656 valid = 0;
657 } else {
658 ferror.clear(f[COL_BRI]);
662 return valid;
665 vlg.dataToView = function(data) {
666 return [data[COL_VID],
667 ((data[COL_MAP].toString() == '') || (data[COL_MAP].toString() == '0')) ? (parseInt(E('_vlan0tag').value) * 1 + data[COL_VID] *1 ).toString() : data[COL_MAP].toString(),
668 (data[COL_P0].toString() != '0') ? 'Yes' : '',
669 (data[COL_P0T].toString() != '0') ? 'On' : '',
670 (data[COL_P1].toString() != '0') ? 'Yes' : '',
671 (data[COL_P1T].toString() != '0') ? 'On' : '',
672 (data[COL_P2].toString() != '0') ? 'Yes' : '',
673 (data[COL_P2T].toString() != '0') ? 'On' : '',
674 (data[COL_P3].toString() != '0') ? 'Yes' : '',
675 (data[COL_P3T].toString() != '0') ? 'On' : '',
676 (data[COL_P4].toString() != '0') ? 'Yes' : '',
677 (data[COL_P4T].toString() != '0') ? 'On' : '',
678 (data[COL_VID_DEF].toString() != '0') ? '*' : '',
679 ['', 'WAN', 'LAN (br0)', 'LAN1 (br1)', 'LAN2 (br2)', 'LAN3 (br3)' ][data[COL_BRI] - 1]];
682 vlg.dataToFieldValues = function (data) {
683 return [data[COL_VID],
684 data[COL_MAP],
685 (data[COL_P0] != 0) ? 'checked' : '',
686 (data[COL_P0T] != 0) ? 'checked' : '',
687 (data[COL_P1] != 0) ? 'checked' : '',
688 (data[COL_P1T] != 0) ? 'checked' : '',
689 (data[COL_P2] != 0) ? 'checked' : '',
690 (data[COL_P2T] != 0) ? 'checked' : '',
691 (data[COL_P3] != 0) ? 'checked' : '',
692 (data[COL_P3T] != 0) ? 'checked' : '',
693 (data[COL_P4] != 0) ? 'checked' : '',
694 (data[COL_P4T] != 0) ? 'checked' : '',
695 (data[COL_VID_DEF] != 0) ? 'checked' : '',
696 data[COL_BRI]];
699 vlg.fieldValuesToData = function(row) {
700 var f = fields.getAll(row);
701 return [f[COL_VID].value,
702 f[COL_MAP].value,
703 f[COL_P0].checked ? 1 : 0,
704 f[COL_P0T].checked ? 1 : 0,
705 f[COL_P1].checked ? 1 : 0,
706 f[COL_P1T].checked ? 1 : 0,
707 f[COL_P2].checked ? 1 : 0,
708 f[COL_P2T].checked ? 1 : 0,
709 f[COL_P3].checked ? 1 : 0,
710 f[COL_P3T].checked ? 1 : 0,
711 f[COL_P4].checked ? 1 : 0,
712 f[COL_P4T].checked ? 1 : 0,
713 f[COL_VID_DEF].checked ? 1 : 0,
714 f[COL_BRI].value];
717 vlg.onCancel = function() {
718 this.removeEditor();
719 this.showSource();
720 this.disableNewEditor(false);
722 this.resetNewEditor();
725 vlg.onAdd = function() {
726 var data;
728 this.moving = null;
729 this.rpHide();
731 if (!this.verifyFields(this.newEditor, false)) return;
733 data = this.fieldValuesToData(this.newEditor);
734 this.insertData(-1, data);
736 this.disableNewEditor(false);
737 this.resetNewEditor();
739 this.resort();
742 vlg.onOK = function() {
743 var i, data, view;
745 if (!this.verifyFields(this.editor, false)) return;
747 data = this.fieldValuesToData(this.editor);
748 view = this.dataToView(data);
750 this.source.setRowData(data);
751 for (i = 0; i < this.source.cells.length; ++i) {
752 this.source.cells[i].innerHTML = view[i];
755 this.removeEditor();
756 this.showSource();
757 this.disableNewEditor(false);
759 this.resetNewEditor();
760 this.resort();
763 vlg.onDelete = function() {
764 this.removeEditor();
765 elem.remove(this.source);
766 this.source = null;
767 this.disableNewEditor(false);
769 this.resetNewEditor();
772 vlg.sortCompare = function(a, b) {
773 var obj = TGO(a);
774 var col = obj.sortColumn;
775 if (this.sortColumn == 0) {
776 var r = cmpInt(parseInt(a.cells[col].innerHTML), parseInt(b.cells[col].innerHTML));
777 } else {
778 var r = cmpText(a.cells[col].innerHTML, b.cells[col].innerHTML);
780 return obj.sortAscending ? r : -r;
783 vlg.resetNewEditor = function() {
784 var f = fields.getAll(this.newEditor);
786 for (var i=0; i <= MAX_BRIDGE_ID; i++) {
787 var j = (i==0) ? '' : i.toString();
788 f[COL_BRI].options[i+2].disabled = (nvram['lan' + j + '_ifname'].length < 1);
791 f[COL_MAP].value = '0';
793 f[COL_VID].selectedIndex=0;
794 var t = MAX_VLAN_ID;
795 while((this.countVID(f[COL_VID].selectedIndex) > 0) && (t > 0)) {
796 f[COL_VID].selectedIndex = (f[COL_VID].selectedIndex%(MAX_VLAN_ID))+1;
797 t--;
800 for(var i=0; i<= MAX_VLAN_ID ; i++) {
801 f[COL_VID].options[i].disabled = (this.countVID(i) > 0);
804 f[COL_P0].checked = 0;
805 f[COL_P0T].checked = 0;
806 f[COL_P0T].disabled = 1;
807 f[COL_P1].checked = 0;
808 f[COL_P1T].checked = 0;
809 f[COL_P1T].disabled = 1;
810 f[COL_P2].checked = 0;
811 f[COL_P2T].checked = 0;
812 f[COL_P2T].disabled = 1;
813 f[COL_P3].checked = 0;
814 f[COL_P3T].checked = 0;
815 f[COL_P3T].disabled = 1;
816 f[COL_P4].checked = 0;
817 f[COL_P4T].checked = 0;
818 f[COL_P4T].disabled = 1;
819 f[COL_VID_DEF].checked = 0;
820 if (this.countDefaultVID()>0)
821 f[COL_VID_DEF].disabled = 1;
822 f[COL_BRI].selectedIndex = 0;
823 ferror.clearAll(fields.getAll(this.newEditor));
825 } // end of the so-called if(supported_device) block
827 function init() {
828 if(port_vlan_supported) {
829 vlg.recolor();
830 vlg.resetNewEditor();
831 var c;
832 if (((c = cookie.get('advanced_vlan_notes_vis')) != null) && (c == '1')) toggleVisibility("notes");
833 if (((c = cookie.get('advanced_vlan_wireless_vis')) != null) && (c == '1')) toggleVisibility("wireless");
834 if (((c = cookie.get('advanced_vlan_vidmap_vis')) != null) && (c == '1')) toggleVisibility("vidmap");
838 function toggleVisibility(whichone) {
839 if (E('sesdiv_' + whichone).style.display == '') {
840 E('sesdiv_' + whichone).style.display = 'none';
841 E('sesdiv_' + whichone + '_showhide').innerHTML = '(Click here to show)';
842 cookie.set('advanced_vlan_' + whichone + '_vis', 0);
843 } else {
844 E('sesdiv_' + whichone).style.display='';
845 E('sesdiv_' + whichone + '_showhide').innerHTML = '(Click here to hide)';
846 cookie.set('advanced_vlan_' + whichone + '_vis', 1);
850 function earlyInit() {
851 if(!port_vlan_supported) {
852 E('save-button').disabled = 1;
853 return;
855 PORT_VLAN_SUPPORT_OVERRIDE = ((nvram['trunk_vlan_so'] == '1') ? 1 : 0);
858 </script>
859 </head>
860 <body onload='init()'>
861 <form id='_fom' method='post' action='tomato.cgi'>
862 <table id='container' cellspacing=0>
863 <tr><td colspan=2 id='header'>
864 <div class='title'>Tomato</div>
865 <div class='version'>Version <% version(); %></div>
866 </td></tr>
867 <tr id='body'><td id='navi'><script type='text/javascript'>navi()</script></td>
868 <td id='content'>
869 <div id='ident'><% ident(); %></div>
870 <input type='hidden' name='_nextpage' value='advanced-vlan.asp'>
871 <input type='hidden' name='_nextwait' value='30'>
872 <input type='hidden' name='_reboot' value='1'>
873 <input type='hidden' name='_nvset' value='1'>
874 <input type='hidden' name='_commit' value='1'>
875 <input type='hidden' name='vlan0ports'>
876 <input type='hidden' name='vlan1ports'>
877 <input type='hidden' name='vlan2ports'>
878 <input type='hidden' name='vlan3ports'>
879 <input type='hidden' name='vlan4ports'>
880 <input type='hidden' name='vlan5ports'>
881 <input type='hidden' name='vlan6ports'>
882 <input type='hidden' name='vlan7ports'>
883 <input type='hidden' name='vlan8ports'>
884 <input type='hidden' name='vlan9ports'>
885 <input type='hidden' name='vlan10ports'>
886 <input type='hidden' name='vlan11ports'>
887 <input type='hidden' name='vlan12ports'>
888 <input type='hidden' name='vlan13ports'>
889 <input type='hidden' name='vlan14ports'>
890 <input type='hidden' name='vlan15ports'>
891 <input type='hidden' name='vlan0hwname'>
892 <input type='hidden' name='vlan1hwname'>
893 <input type='hidden' name='vlan2hwname'>
894 <input type='hidden' name='vlan3hwname'>
895 <input type='hidden' name='vlan4hwname'>
896 <input type='hidden' name='vlan5hwname'>
897 <input type='hidden' name='vlan6hwname'>
898 <input type='hidden' name='vlan7hwname'>
899 <input type='hidden' name='vlan8hwname'>
900 <input type='hidden' name='vlan9hwname'>
901 <input type='hidden' name='vlan10hwname'>
902 <input type='hidden' name='vlan11hwname'>
903 <input type='hidden' name='vlan12hwname'>
904 <input type='hidden' name='vlan13hwname'>
905 <input type='hidden' name='vlan14hwname'>
906 <input type='hidden' name='vlan15hwname'>
907 <input type='hidden' name='wan_ifnameX'>
908 <input type='hidden' name='manual_boot_nv'>
909 <input type='hidden' name='lan_ifnames'>
910 <input type='hidden' name='lan1_ifnames'>
911 <input type='hidden' name='lan2_ifnames'>
912 <input type='hidden' name='lan3_ifnames'>
913 <input type='hidden' name='trunk_vlan_so'>
914 <input type='hidden' name='vlan0vid'>
915 <input type='hidden' name='vlan1vid'>
916 <input type='hidden' name='vlan2vid'>
917 <input type='hidden' name='vlan3vid'>
918 <input type='hidden' name='vlan4vid'>
919 <input type='hidden' name='vlan5vid'>
920 <input type='hidden' name='vlan6vid'>
921 <input type='hidden' name='vlan7vid'>
922 <input type='hidden' name='vlan8vid'>
923 <input type='hidden' name='vlan9vid'>
924 <input type='hidden' name='vlan10vid'>
925 <input type='hidden' name='vlan11vid'>
926 <input type='hidden' name='vlan12vid'>
927 <input type='hidden' name='vlan13vid'>
928 <input type='hidden' name='vlan14vid'>
929 <input type='hidden' name='vlan15vid'>
931 <div id='sesdiv' style='display:none'>
932 <div class='section-title'>VLAN</div>
933 <div class='section'>
934 <table class='tomato-grid' cellspacing=1 id='vlan-grid'></table>
935 </div>
937 <!-- / / / -->
939 <div class='section-title'>VID Offset <small><i><a href='javascript:toggleVisibility("vidmap");'><span id='sesdiv_vidmap_showhide'>(Click here to show)</span></a></i></small></div>
940 <div class='section' id='sesdiv_vidmap' style='display:none'>
941 <script type='text/javascript'>
942 createFieldTable('', [
943 { title: 'First 802.1Q VLAN tag', name: 'vlan0tag', type: 'text', maxlen:4, size:6,
944 value: fixInt(nvram.vlan0tag, 0, 4080, 0),
945 suffix: ' <small><i>(range: 0 - 4080; must be a multiple of 16; set to 0 to disable)</i></small>' }
947 </script>
948 </div>
950 <!-- / / / -->
952 <div class='section-title'>Wireless <small><i><a href='javascript:toggleVisibility("wireless");'><span id='sesdiv_wireless_showhide'>(Click here to show)</span></a></i></small></div>
953 <div class='section' id='sesdiv_wireless' style='display:none'>
954 <script type='text/javascript'>
955 var f = [];
956 for (var uidx = 0; uidx < wl_ifaces.length; ++uidx) {
957 var u = wl_fface(uidx);
958 f.push(
959 { title: ('Bridge ' + wl_ifaces[uidx][0] + ' to'), name: ('f_bridge_wlan'+u+'_to'), type: 'select',
960 options: [[0,'LAN (br0)'],[1,'LAN1 (br1)'],[2,'LAN2 (br2)'],[3,'LAN3 (br3)'],[4,'none']], value: 4 } );
962 createFieldTable('',f);
963 if(port_vlan_supported) vlg.setup();
964 </script>
965 </div>
967 <!-- / / / -->
969 <div class='section-title'>Notes <small><i><a href='javascript:toggleVisibility("notes");'><span id='sesdiv_notes_showhide'>(Click here to show)</span></a></i></small></div>
970 <div class='section' id='sesdiv_notes' style='display:none'>
971 <ul>
972 <li><b>VLAN</b> - Unique identifier of a VLAN.</li>
973 <li><b>VID</b> - <i>EXPERIMENTAL</i> - Allows overriding 'traditional' VLAN/VID mapping with arbitrary VIDs for each VLAN (set to '0' to use 'regular' VLAN/VID mappings instead). Warning: this hasn't been verified/tested on anything but a Cisco/Linksys E3000 and may not be supported by your particular device/model (<small><b><i>see notes on "VID Offset" below</i></b></small>).</li>
974 <li><b>Ports 1-4 &amp; WAN</b> - Which ethernet ports on the router should be members of this VLAN.</li>
975 <li><b>Tagged</b> - Enable 802.1Q tagging of ethernet frames on a particular port/VLAN
976 <script type='text/javascript'>
977 if(!trunk_vlan_supported)
978 W(' <i><b>(unknown support for this model...contact the developper (Victek))</i></b>');
979 </script>
980 </li>
981 <li><b>Default</b> - VLAN ID assigned to untagged frames received by the router.</li>
982 <li><b>Bridge</b> - Determines if this VLAN ID should be treated as WAN, part of a LAN bridge or just left alone (i.e. member of a 802.1Q trunk, being managed manually via scripts, etc...).</li>
983 </ul>
985 <ul>
986 <li><b>VID Offset</b> - <i>EXPERIMENTAL</i> - First 802.1Q VLAN tag to be used as <i>base/initial tag/VID</i> for VLAN and VID assignments. This allows using VIDs larger than 15 on (older) devices such as the Linksys WRT54GL v1.1 (in contiguous blocks/ranges with up to 16 VLANs/VIDs). Set to '0' (zero) to disable this feature and VLANs will have the very same/identical value for its VID, as usual (from 0 to 15).</li>
987 </ul>
989 <ul>
990 <li><b>Wireless</b> - Assignments of wireless interfaces to different LAN briges. You should probably be using and/or check things on <a href=advanced-wlanvifs.asp>Advanced/Virtual Wireless</a> and <a href=basic-network.asp>Basic/Network</a>.</li>
991 </ul>
993 <small>
994 <ul>
995 <li><b>Other relevant notes/hints:</b>
996 <ul>
997 <li>One VID <i>must</i> be assigned to WAN.</li>
998 <li>One VID <i>must</i> be selected as the default.</li>
999 <script type='text/javascript'>
1000 if((trunk_vlan_supported) || (nvram.trunk_vlan_so == '1')) {
1001 W('<li>To prevent 802.1Q compatibility issues, avoid using VID "0" as 802.1Q specifies that frames with a tag of "0" do not belong to any VLAN (the tag contains only user priority information).</li>');
1002 W('<li>It may be also recommended to avoid using VID "1" as some vendors consider it special/reserved (for management purposes).</li>');
1004 </script>
1005 </ul>
1006 <br>
1007 <ul>
1008 <li>This is an <b>experimental</b> feature and hasn't been tested in anything but a Linksys WRT54GL v1.1 running a Teaman-ND K24 build and a Cisco/Linksys E3000 running a Teaman-RT K26 build.</li>
1009 <li>There's lots of things that could go wrong, please do think about what you're doing and take a backup before hitting the 'Save' button on this page!</li>
1010 </ul>
1011 </ul>
1012 <div id='trunk_vlan_override' style='display:none'>
1013 <div class='section-title'>Trunk VLAN support override (experimental)</div>
1014 <div class='section'>
1015 <script type='text/javascript'>
1016 createFieldTable('', [
1017 { title: 'Enable', name: 'f_trunk_vlan_so', type: 'checkbox', value: nvram.trunk_vlan_so == '1' },
1019 </script>
1020 </div>
1021 </div>
1022 </div>
1023 </small>
1024 </div>
1025 </div>
1026 <script type='text/javascript'>
1027 if(!port_vlan_supported)
1028 W('<i>This feature is not supported on this router.</i>');
1029 else {
1030 E('sesdiv').style.display = '';
1031 if(!trunk_vlan_supported)
1032 E('trunk_vlan_override').style.display = '';
1034 </script>
1035 </td></tr>
1036 <tr><td id='footer' colspan=2>
1037 <span id='footer-msg'></span>
1038 <input type='button' value='Save' id='save-button' onclick='save()'>
1039 <input type='button' value='Cancel' id='cancel-button' onclick='javascript:reloadPage();'>
1040 </td></tr>
1041 </table>
1042 </form>
1043 <script type='text/javascript'>earlyInit(); verifyFields(null,1);</script>
1044 </body>
1045 </html>