ports and addresses must be malloced to avoid segfault on exit

This commit is contained in:
Matt Johnston 2015-12-02 22:37:20 +08:00
parent f64a50f319
commit e81b6fbc6e
2 changed files with 24 additions and 21 deletions

View File

@ -72,7 +72,8 @@ typedef struct svr_runopts {
int forkbg; int forkbg;
int usingsyslog; int usingsyslog;
/* ports is an array of the portcount listening ports */ /* ports and addresses are arrays of the portcount
listening ports. strings are malloced. */
char *ports[DROPBEAR_MAX_PORTS]; char *ports[DROPBEAR_MAX_PORTS];
unsigned int portcount; unsigned int portcount;
char *addresses[DROPBEAR_MAX_PORTS]; char *addresses[DROPBEAR_MAX_PORTS];

View File

@ -33,7 +33,7 @@
svr_runopts svr_opts; /* GLOBAL */ svr_runopts svr_opts; /* GLOBAL */
static void printhelp(const char * progname); static void printhelp(const char * progname);
static void addportandaddress(char* spec); static void addportandaddress(const char* spec);
static void loadhostkey(const char *keyfile, int fatal_duplicate); static void loadhostkey(const char *keyfile, int fatal_duplicate);
static void addhostkey(const char *keyfile); static void addhostkey(const char *keyfile);
@ -348,54 +348,56 @@ void svr_getopts(int argc, char ** argv) {
} }
} }
static void addportandaddress(char* spec) { static void addportandaddress(const char* spec) {
char *spec_copy = NULL, *myspec = NULL, *port = NULL, *address = NULL;
char *myspec = NULL;
if (svr_opts.portcount < DROPBEAR_MAX_PORTS) { if (svr_opts.portcount < DROPBEAR_MAX_PORTS) {
/* We don't free it, it becomes part of the runopt state */ /* We don't free it, it becomes part of the runopt state */
myspec = m_strdup(spec); spec_copy = m_strdup(spec);
myspec = spec_copy;
if (myspec[0] == '[') { if (myspec[0] == '[') {
myspec++; myspec++;
svr_opts.ports[svr_opts.portcount] = strchr(myspec, ']'); port = strchr(myspec, ']');
if (svr_opts.ports[svr_opts.portcount] == NULL) { if (!port) {
/* Unmatched [ -> exit */ /* Unmatched [ -> exit */
dropbear_exit("Bad listen address"); dropbear_exit("Bad listen address");
} }
svr_opts.ports[svr_opts.portcount][0] = '\0'; port[0] = '\0';
svr_opts.ports[svr_opts.portcount]++; port++;
if (svr_opts.ports[svr_opts.portcount][0] != ':') { if (port[0] != ':') {
/* Missing port -> exit */ /* Missing port -> exit */
dropbear_exit("Missing port"); dropbear_exit("Missing port");
} }
} else { } else {
/* search for ':', that separates address and port */ /* search for ':', that separates address and port */
svr_opts.ports[svr_opts.portcount] = strrchr(myspec, ':'); port = strrchr(myspec, ':');
} }
if (svr_opts.ports[svr_opts.portcount] == NULL) { if (!port) {
/* no ':' -> the whole string specifies just a port */ /* no ':' -> the whole string specifies just a port */
svr_opts.ports[svr_opts.portcount] = myspec; port = myspec;
} else { } else {
/* Split the address/port */ /* Split the address/port */
svr_opts.ports[svr_opts.portcount][0] = '\0'; port[0] = '\0';
svr_opts.ports[svr_opts.portcount]++; port++;
svr_opts.addresses[svr_opts.portcount] = myspec; address = myspec;
} }
if (svr_opts.addresses[svr_opts.portcount] == NULL) { if (!address) {
/* no address given -> fill in the default address */ /* no address given -> fill in the default address */
svr_opts.addresses[svr_opts.portcount] = m_strdup(DROPBEAR_DEFADDRESS); address = DROPBEAR_DEFADDRESS;
} }
if (svr_opts.ports[svr_opts.portcount][0] == '\0') { if (port[0] == '\0') {
/* empty port -> exit */ /* empty port -> exit */
dropbear_exit("Bad port"); dropbear_exit("Bad port");
} }
svr_opts.ports[svr_opts.portcount] = m_strdup(port);
svr_opts.addresses[svr_opts.portcount] = m_strdup(address);
svr_opts.portcount++; svr_opts.portcount++;
m_free(spec_copy);
} }
} }