/* SmoothWall libsmooth. * * This program is distributed under the terms of the GNU General Public * Licence. See the file COPYING for details. * * (c) Lawrence Manning, 2001 * Contains network library functions. * */ #include #include // Translation #include #define _(x) dgettext("setup", x) #include "setup.h" extern FILE *flog; extern char *mylog; extern struct nic nics[]; extern struct knic knics[]; char *ucolourcard[] = { "GREEN", "RED", "ORANGE", "BLUE", NULL }; char *lcolourcard[] = { "green", "red", "orange", "blue", NULL }; int scanned_nics_read_done = 0; newtComponent networkform; newtComponent addressentry; newtComponent netmaskentry; newtComponent gatewayentry; newtComponent statictyperadio; newtComponent dhcptyperadio; newtComponent pppoetyperadio; newtComponent dhcphostnameentry; newtComponent dhcpforcemtuentry; newtComponent dhcprapidcommitentry; /* acceptable character filter for IP and netmaks entry boxes */ static int ip_input_filter(newtComponent entry, void * data, int ch, int cursor) { if ((ch >= '0' && ch <= '9') || ch == '.' || ch == '\r' || ch >= NEWT_KEY_EXTRA_BASE) return ch; return 0; } /* This is a groovie dialog for showing network info. Takes a keyvalue list, * a colour and a dhcp flag. Shows the current settings, and rewrites them * if necessary. DHCP flag sets wether to show the dhcp checkbox. */ int changeaddress(struct keyvalue *kv, char *colour, int typeflag, char *defaultdhcphostname) { const char *addressresult; const char *netmaskresult; const char *gatewayresult; const char *dhcphostnameresult; const char *dhcpforcemturesult; struct newtExitStruct es; newtComponent header; newtComponent addresslabel; newtComponent netmasklabel; newtComponent gatewaylabel; newtComponent dhcphostnamelabel; newtComponent dhcpforcemtulabel; newtComponent dhcprapidcommitlabel; newtComponent ok, cancel; char message[1000]; char temp[STRING_SIZE]; char addressfield[STRING_SIZE]; char netmaskfield[STRING_SIZE]; char gatewayfield[STRING_SIZE]; char typefield[STRING_SIZE]; char dhcphostnamefield[STRING_SIZE]; char dhcpforcemtufield[STRING_SIZE]; char dhcprapidcommitfield[STRING_SIZE]; char enablerapidcommit; int error; int result = 0; char type[STRING_SIZE]; int startstatictype = 0; int startdhcptype = 0; int startpppoetype = 0; unsigned char buffer[sizeof(struct in_addr)]; /* Build some key strings. */ sprintf(addressfield, "%s_ADDRESS", colour); sprintf(netmaskfield, "%s_NETMASK", colour); sprintf(gatewayfield, "DEFAULT_GATEWAY"); sprintf(typefield, "%s_TYPE", colour); sprintf(dhcphostnamefield, "%s_DHCP_HOSTNAME", colour); sprintf(dhcpforcemtufield, "%s_DHCP_FORCE_MTU", colour); sprintf(dhcprapidcommitfield, "%s_DHCP_RAPID_COMMIT", colour); sprintf(message, _("Interface - %s"), colour); newtCenteredWindow(44, (typeflag ? 20 : 12), message); networkform = newtForm(NULL, NULL, 0); sprintf(message, _("Enter the IP address information for the %s interface."), colour); header = newtTextboxReflowed(1, 1, message, 42, 0, 0, 0); newtFormAddComponent(networkform, header); /* See if we need a dhcp checkbox. If we do, then we shift the contents * of the window down two rows to make room. */ if (typeflag) { *temp = '\0'; // Find RapidCommit setting findkey(kv, dhcprapidcommitfield, temp); if (strcmp(temp, "yes") == 0 || strcmp(temp, "true") == 0 || strcmp(temp, "on") == 0 || strcmp(temp, "") == 0) enablerapidcommit = '*'; else enablerapidcommit = ' '; strcpy(temp, "STATIC"); findkey(kv, typefield, temp); if (strcmp(temp, "STATIC") == 0) startstatictype = 1; if (strcmp(temp, "DHCP") == 0) startdhcptype = 1; if (strcmp(temp, "PPPOE") == 0) startpppoetype = 1; statictyperadio = newtRadiobutton(2, 4, _("Static"), startstatictype, NULL); dhcptyperadio = newtRadiobutton(2, 5, _("DHCP"), startdhcptype, statictyperadio); pppoetyperadio = newtRadiobutton(2, 6, _("PPP DIALUP (PPPoE, modem, ATM ...)"), startpppoetype, dhcptyperadio); newtFormAddComponents(networkform, statictyperadio, dhcptyperadio, pppoetyperadio, NULL); newtComponentAddCallback(statictyperadio, networkdialogcallbacktype, NULL); newtComponentAddCallback(dhcptyperadio, networkdialogcallbacktype, NULL); newtComponentAddCallback(pppoetyperadio, networkdialogcallbacktype, NULL); dhcphostnamelabel = newtTextbox(2, 8, 18, 1, 0); newtTextboxSetText(dhcphostnamelabel, _("DHCP Hostname:")); dhcpforcemtulabel = newtTextbox(2, 9, 18, 1, 0); newtTextboxSetText(dhcpforcemtulabel, _("Force DHCP MTU:")); dhcprapidcommitlabel = newtTextbox(2, 10, 18, 1, 0); newtTextboxSetText(dhcprapidcommitlabel, _("Rapid Commit:")); strcpy(temp, defaultdhcphostname); findkey(kv, dhcphostnamefield, temp); dhcphostnameentry = newtEntry(20, 8, temp, 20, &dhcphostnameresult, 0); strcpy(temp, ""); findkey(kv, dhcpforcemtufield, temp); dhcpforcemtuentry = newtEntry(20, 9, temp, 20, &dhcpforcemturesult, 0); dhcprapidcommitentry = newtCheckbox(20, 10, "", enablerapidcommit, " *", &enablerapidcommit); newtComponentAddCallback(dhcprapidcommitentry, networkdialogcallbacktype, NULL); newtFormAddComponent(networkform, dhcphostnamelabel); newtFormAddComponent(networkform, dhcphostnameentry); newtFormAddComponent(networkform, dhcpforcemtulabel); newtFormAddComponent(networkform, dhcpforcemtuentry); newtFormAddComponent(networkform, dhcprapidcommitlabel); newtFormAddComponent(networkform, dhcprapidcommitentry); if (startdhcptype == 0) { newtEntrySetFlags(dhcphostnameentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtEntrySetFlags(dhcpforcemtuentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtCheckboxSetFlags(dhcprapidcommitentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); } } /* Address */ addresslabel = newtTextbox(2, (typeflag ? 12 : 4) + 0, 18, 1, 0); newtTextboxSetText(addresslabel, _("IP address:")); strcpy(temp, ""); findkey(kv, addressfield, temp); addressentry = newtEntry(20, (typeflag ? 12 : 4) + 0, temp, 20, &addressresult, 0); newtEntrySetFilter(addressentry, ip_input_filter, NULL); if (typeflag == 1 && startstatictype == 0) newtEntrySetFlags(addressentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtFormAddComponent(networkform, addresslabel); newtFormAddComponent(networkform, addressentry); /* Netmask */ netmasklabel = newtTextbox(2, (typeflag ? 12 : 4) + 1, 18, 1, 0); newtTextboxSetText(netmasklabel, _("Network mask:")); strcpy(temp, "255.255.255.0"); findkey(kv, netmaskfield, temp); netmaskentry = newtEntry(20, (typeflag ? 12 : 4) + 1, temp, 20, &netmaskresult, 0); newtEntrySetFilter(netmaskentry, ip_input_filter, NULL); if (typeflag == 1 && startstatictype == 0) newtEntrySetFlags(netmaskentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtFormAddComponent(networkform, netmasklabel); newtFormAddComponent(networkform, netmaskentry); if (typeflag) { /* Gateway */ gatewaylabel = newtTextbox(2, (typeflag ? 12 : 4) + 2, 18, 1, 0); newtTextboxSetText(gatewaylabel, _("Gateway:")); strcpy(temp, ""); findkey(kv, gatewayfield, temp); gatewayentry = newtEntry(20, (typeflag ? 12 : 4) + 2, temp, 20, &gatewayresult, 0); newtEntrySetFilter(gatewayentry, ip_input_filter, NULL); if (typeflag == 1 && startstatictype == 0) newtEntrySetFlags(gatewayentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtFormAddComponent(networkform, gatewaylabel); newtFormAddComponent(networkform, gatewayentry); } /* Buttons. */ ok = newtButton(8, (typeflag ? 16 : 7), _("OK")); cancel = newtButton(26, (typeflag ? 16 : 7), _("Cancel")); newtFormAddComponents(networkform, ok, cancel, NULL); newtRefresh(); newtDrawForm(networkform); do { error = 0; newtFormRun(networkform, &es); if (es.u.co == ok) { /* OK was pressed; verify the contents of each entry. */ strcpy(message, _("The following fields are invalid:")); strcat(message, "\n\n"); strcpy(type, "STATIC"); if (typeflag) gettype(type); if (strcmp(type, "STATIC") == 0) { if (inet_addr(addressresult) == INADDR_NONE) { strcat(message, _("IP address")); strcat(message, "\n"); error = 1; } if (inet_pton(AF_INET, netmaskresult, &buffer) == 0) { strcat(message, _("Network mask")); strcat(message, "\n"); error = 1; } if (typeflag && (inet_addr(gatewayresult) == INADDR_NONE)) { strcat(message, _("Gateway")); strcat(message, "\n"); error = 1; } } if (strcmp(type, "DHCP") == 0) { if (!strlen(dhcphostnameresult)) { strcat(message, _("DHCP hostname")); strcat(message, "\n"); error = 1; } } if (error) errorbox(message); else { /* No errors! Set new values, depending on dhcp flag etc. */ if (typeflag) { replacekeyvalue(kv, dhcphostnamefield, dhcphostnameresult); replacekeyvalue(kv, dhcpforcemtufield, dhcpforcemturesult); if (enablerapidcommit == '*') replacekeyvalue(kv, dhcprapidcommitfield, "on"); else replacekeyvalue(kv, dhcprapidcommitfield, "off"); if (strcmp(type, "STATIC") != 0) { replacekeyvalue(kv, addressfield, "0.0.0.0"); replacekeyvalue(kv, netmaskfield, "0.0.0.0"); replacekeyvalue(kv, gatewayfield, "0.0.0.0"); } else { replacekeyvalue(kv, addressfield, addressresult); replacekeyvalue(kv, netmaskfield, netmaskresult); replacekeyvalue(kv, gatewayfield, gatewayresult); } replacekeyvalue(kv, typefield, type); } else { replacekeyvalue(kv, addressfield, addressresult); replacekeyvalue(kv, netmaskfield, netmaskresult); } setnetaddress(kv, colour); result = 1; } } /* Workaround for a bug that dhcp radiobutton also end the dialog at arm */ else { if (es.u.co != cancel) { error = 1; } } } while (error); newtFormDestroy(networkform); newtPopWindow(); return result; } /* for pppoe: return string thats type STATIC, DHCP or PPPOE */ int gettype(char *type) { newtComponent selected = newtRadioGetCurrent(statictyperadio); if (selected == statictyperadio) strcpy(type, "STATIC"); else if (selected == dhcptyperadio) strcpy(type, "DHCP"); else if (selected == pppoetyperadio) strcpy(type, "PPPOE"); else strcpy(type, "ERROR"); return 0; } int setnetaddress(struct keyvalue *kv, char *colour) { char addressfield[STRING_SIZE]; char netaddressfield[STRING_SIZE]; char netmaskfield[STRING_SIZE]; char address[STRING_SIZE]; char netmask[STRING_SIZE]; unsigned long int intaddress; unsigned long int intnetaddress; unsigned long int intnetmask; struct in_addr temp; char *netaddress; /* Build some key strings. */ sprintf(addressfield, "%s_ADDRESS", colour); sprintf(netaddressfield, "%s_NETADDRESS", colour); sprintf(netmaskfield, "%s_NETMASK", colour); strcpy(address, ""); findkey(kv, addressfield, address); strcpy(netmask, ""); findkey(kv, netmaskfield, netmask); /* Calculate netaddress. Messy.. */ intaddress = inet_addr(address); intnetmask = inet_addr(netmask); intnetaddress = intaddress & intnetmask; temp.s_addr = intnetaddress; netaddress = inet_ntoa(temp); replacekeyvalue(kv, netaddressfield, netaddress); return 1; } /* Called when dhcp flag is toggled. Toggle disabled state of other 3 * controls. */ void networkdialogcallbacktype(newtComponent cm, void *data) { char type[STRING_SIZE]; gettype(type); if (strcmp(type, "STATIC") != 0) { newtEntrySetFlags(addressentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtEntrySetFlags(netmaskentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtEntrySetFlags(gatewayentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); } else { newtEntrySetFlags(addressentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_RESET); newtEntrySetFlags(netmaskentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_RESET); newtEntrySetFlags(gatewayentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_RESET); } if (strcmp(type, "DHCP") == 0) { newtEntrySetFlags(dhcphostnameentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_RESET); newtEntrySetFlags(dhcpforcemtuentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_RESET); newtCheckboxSetFlags(dhcprapidcommitentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_RESET); } else { newtEntrySetFlags(dhcphostnameentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtEntrySetFlags(dhcpforcemtuentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); newtCheckboxSetFlags(dhcprapidcommitentry, NEWT_FLAG_DISABLED, NEWT_FLAGS_SET); } newtRefresh(); newtDrawForm(networkform); } int interfacecheck(struct keyvalue *kv, char *colour) { char temp[STRING_SIZE]; char colourfields[NETCHANGE_TOTAL][STRING_SIZE]; int c; sprintf(colourfields[ADDRESS], "%s_ADDRESS", colour); sprintf(colourfields[NETADDRESS], "%s_NETADDRESS", colour); sprintf(colourfields[NETMASK], "%s_NETMASK", colour); for (c = 0; c < 3; c++) { strcpy(temp, ""); findkey(kv, colourfields[c], temp); if (!(strlen(temp))) return 0; } return 1; } /* Funky routine for loading all drivers (cept those are already loaded.). */ int probecards(char *driver, char *driveroptions ) { return 0; } int get_knic(int card) //returns "0" for zero cards or error and "1" card is found. { struct keyvalue *kv = initkeyvalues(); char temp[STRING_SIZE], searchstr[STRING_SIZE]; int ret_value; if (!(readkeyvalues(kv, CONFIG_ROOT "/ethernet/settings"))) { freekeyvalues(kv); errorbox(_("Unable to open settings file")); return 0; } sprintf(searchstr, "%s_MACADDR", ucolourcard[card]); strcpy(temp, ""); findkey(kv, searchstr, temp); if (strlen(temp)) { strcpy(knics[ card ].macaddr, temp); strcpy(knics[ card ].colour, ucolourcard[card]); sprintf(searchstr, "%s_DESCRIPTION", ucolourcard[card]); findkey(kv, searchstr, temp); strcpy(knics[ card ].description, temp); sprintf(searchstr, "%s_DRIVER", ucolourcard[card]); findkey(kv, searchstr, temp); strcpy(knics[ card ].driver, temp); ret_value = 1; } else { strcpy(knics[ card ].description, _("Unset")); ret_value = 0; } freekeyvalues(kv); return ret_value; } int init_knics(void) { int found = 0; found += get_knic(_GREEN_CARD_); found += get_knic(_RED_CARD_); found += get_knic(_ORANGE_CARD_); found += get_knic(_BLUE_CARD_); return found; } int fmt_exists(const char *fname) { /* Check if it is any file or directory */ struct stat st; if (stat(fname, &st) == -1) return 0; else return 1; } int is_interface_up(char *card) { /* Check if the interface is UP */ char temp[STRING_SIZE]; sprintf(temp,"ip link show dev %s | grep -q UP", card); if (mysystem(NULL, temp)) return 0; else return 1; } int rename_device(char *old_name, char *new_name) { char temp[STRING_SIZE]; sprintf(temp,SYSDIR "/%s", old_name); if (!(fmt_exists(temp))) { fprintf(flog,"Device not found: %s\n",old_name); return 0; } sprintf(temp,"/sbin/ip link set dev %s name %s",old_name ,new_name ); mysystem(NULL, temp); return 1; } char g_temp[STRING_SIZE]=""; char* readmac(char *card) { FILE *fp; char temp[STRING_SIZE], mac[20]; sprintf(temp,"/sys/class/net/%s/address",card); if( (fp = fopen(temp, "r")) == NULL ) { fprintf(flog,"Couldn't open: %s\n",temp); return NULL; } fgets(mac, 18, fp); strtok(mac,"\n"); fclose(fp); strcpy(g_temp, mac); return g_temp; } char* find_nic4mac(char *findmac) { DIR *dir; struct dirent *dirzeiger; char temp[STRING_SIZE] = ""; char temp2[STRING_SIZE] = ""; if((dir=opendir(SYSDIR)) == NULL) { fprintf(flog,"Fehler bei opendir (find_name4nic) ...\n"); return NULL; } while((dirzeiger=readdir(dir)) != NULL) { if(*((*dirzeiger).d_name) != '.' & strcmp(((*dirzeiger).d_name), "lo") != 0) { sprintf(temp2, "%s", readmac((*dirzeiger).d_name) ); if (strcmp(findmac, temp2) == 0) { sprintf(temp,"%s", (*dirzeiger).d_name); break; } } } if(closedir(dir) == -1) fprintf(flog,"Fehler beim schliessen von %s\n", SYSDIR); strcpy(g_temp, temp); return g_temp; } int nic_shutdown(char *nic) { char temp[STRING_SIZE]; sprintf(temp,"ip link set %s down", nic); mysystem(NULL, temp); } int nic_startup(char *nic) { char temp[STRING_SIZE]; sprintf(temp,"ip link set %s up", nic); mysystem(NULL, temp); } int rename_nics(void) { int i, j, k; int fnics = scan_network_cards(); char nic2find[STRING_SIZE], temp[STRING_SIZE]; for(i=0; i<4; i++) if (strcmp(knics[i].macaddr, "")) for(j=0; j 15 ) { sprintf(nics[count].nic, "%s", find_nic4mac(nics[count].macaddr)); count++; } } fclose(fp); scanned_nics_read_done = count; } else fprintf(flog,"Scan Networkcards does read.\n"); return scanned_nics_read_done; } int nicmenu(int colour) { int rc, choise = 0, count = 0, kcount = 0, mcount = 0, i, j, nic_in_use; int found_NIC_as_Card[MAX_NICS]; char message[STRING_SIZE]; char temp[STRING_SIZE]; char cMenuInhalt[STRING_SIZE]; char MenuInhalt[MAX_NICS][STRING_SIZE]; char *pMenuInhalt[MAX_NICS]; while (strcmp(nics[count].macaddr, "")) count++; // 2 find how many nics in system for (i=0; i kcount) { for (i=0 ; i < count ; i++) { nic_in_use = 0; for (j=0 ; j <= kcount ; j++) { if (strcmp(nics[ i ].macaddr, knics[ j ].macaddr) == 0 ) { nic_in_use = 1; break; } } if (!(nic_in_use)) { if ( strlen(nics[i].description) < 55 ) sprintf(MenuInhalt[mcount], "%.*s", strlen(nics[i].description)-2, nics[i].description+1); else { sprintf(cMenuInhalt, "%.50s", nics[i].description + 1); sprintf(MenuInhalt[mcount], cMenuInhalt); strcat (MenuInhalt[mcount], "..."); } while ( strlen(MenuInhalt[mcount]) < 53) strcat(MenuInhalt[mcount], " "); // Fill with space. strcat(MenuInhalt[mcount], " ("); strcat(MenuInhalt[mcount], nics[i].macaddr); strcat(MenuInhalt[mcount], ")"); pMenuInhalt[mcount] = MenuInhalt[mcount]; found_NIC_as_Card[mcount]=i; mcount++; } } pMenuInhalt[mcount] = NULL; sprintf(message, _("Please choose a networkcard for the following interface - %s."), ucolourcard[colour]); rc=2; while ( rc == 2 ) { rc = newtWinMenu(_("Extended Network Menu"), message, 50, 5, 5, mcount, pMenuInhalt, &choise, _("Select"), _("Identify"), _("Cancel"), NULL); if ( rc == 2 ) { sprintf(temp, "/sbin/ip link set %s up", nics[found_NIC_as_Card[choise]].nic); mysystem(NULL, temp); sprintf(temp, "/usr/sbin/ethtool -p %s 10", nics[found_NIC_as_Card[choise]].nic); if (runcommandwithstatus(temp, _("Device Identification"), _("The lights on the selected port should flash now for 10 seconds..."), NULL) != 0) { errorbox(_("Identification is not supported by this interface.")); sprintf(temp, "/sbin/ip link set %s down", nics[found_NIC_as_Card[choise]].nic); mysystem(NULL, temp); } } } if ( rc == 0 || rc == 1) { write_configs_netudev(found_NIC_as_Card[choise], colour); } return 0; } else { // We have to add here that you can manually add a device errorbox(_("There are no unassigned interfaces on your system.")); return 1; } } int clear_card_entry(int card) { struct keyvalue *kv = initkeyvalues(); char temp[STRING_SIZE]; if (!(readkeyvalues(kv, CONFIG_ROOT "/ethernet/settings"))) { freekeyvalues(kv); errorbox(_("Unable to open settings file")); return 0; } strcpy(knics[card].driver, ""); strcpy(knics[card].description, _("Unset")); strcpy(knics[card].macaddr, ""); strcpy(knics[card].colour, ""); sprintf(temp, "%s_DRIVER", ucolourcard[card]); replacekeyvalue(kv, temp, ""); sprintf(temp, "%s_DEV", ucolourcard[card]); replacekeyvalue(kv, temp, ""); sprintf(temp, "%s_MACADDR", ucolourcard[card]); replacekeyvalue(kv, temp, ""); sprintf(temp, "%s_DESCRIPTION", ucolourcard[card]); replacekeyvalue(kv, temp, ""); writekeyvalues(kv, CONFIG_ROOT "/ethernet/settings"); freekeyvalues(kv); return 0; } int ask_clear_card_entry(int card) { char message[STRING_SIZE]; int rc; sprintf(message, _("Do you really want to remove the assigned %s interface?"), ucolourcard[card]); rc = newtWinChoice(_("Warning"), _("OK"), _("Cancel"), message); if ( rc == 0 || rc == 1) { clear_card_entry(card); } else return 1; return 0; } /* Manual entry for gurus. */ int manualdriver(char *driver, char *driveroptions) { char *values[] = { NULL, NULL }; /* pointers for the values. */ struct newtWinEntry entries[] = { { "", &values[0], 0,}, { NULL, NULL, 0 } }; int rc; char commandstring[STRING_SIZE]; char *driverend; strcpy(driver, ""); strcpy(driveroptions, ""); rc = newtWinEntries(_("Select network driver"), _("Set additional module parameters"), 50, 5, 5, 40, entries, _("OK"), _("Cancel"), NULL); if (rc == 0 || rc == 1) { if (strlen(values[0])) { sprintf(commandstring, "/sbin/modprobe %s", values[0]); if (runcommandwithstatus(commandstring, _("Loading module..."), _("Loading module..."), NULL) == 0) { if ((driverend = strchr(values[0], ' '))) { *driverend = '\0'; strcpy(driver, values[0]); strcpy(driveroptions, driverend + 1); } else { strcpy(driver, values[0]); strcpy(driveroptions, ""); } } else errorbox(_("Unable to load driver module.")); } else errorbox(_("Module name cannot be blank.")); } free(values[0]); return 1; }