nuttx-apps/examples/webserver/webserver_main.c

233 lines
6.7 KiB
C
Raw Normal View History

/****************************************************************************
* apps/examples/webserver/webserver_main.c
*
* Copyright (C) 2007, 2009-2012, 2015 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <gnutt@nuttx.org>
*
* Based on uIP which also has a BSD style license:
*
* Copyright (c) 2001, Adam Dunkels.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by Adam Dunkels.
* 4. The name of the author may not be used to endorse or promote
* products derived from this software without specific prior
* written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
* GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <sys/ioctl.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <time.h>
#include <debug.h>
#include <net/if.h>
#include <netinet/in.h>
#include <nuttx/net/arp.h>
#include "netutils/netlib.h"
#ifdef CONFIG_EXAMPLES_WEBSERVER_DHCPC
#include <arpa/inet.h>
#endif
/* Here we include the header file for the application(s) we use in
* our project as defined in the config/<board-name>/defconfig file
*/
/* DHCPC may be used in conjunction with any other feature (or not) */
#ifdef CONFIG_EXAMPLES_WEBSERVER_DHCPC
# include "netutils/dhcpc.h"
#endif
/* Include uIP webserver definitions */
#include "netutils/httpd.h"
#include "cgi.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
/****************************************************************************
* Private Data
****************************************************************************/
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* webserver_main
****************************************************************************/
int main(int argc, FAR char *argv[])
{
#ifndef CONFIG_NSH_NETINIT
/* We are running standalone (as opposed to a NSH built-in app). Therefore
* we need to initialize the network before we start.
*/
struct in_addr addr;
#if defined(CONFIG_EXAMPLES_WEBSERVER_DHCPC) || defined(CONFIG_EXAMPLES_WEBSERVER_NOMAC)
uint8_t mac[IFHWADDRLEN];
#endif
#ifdef CONFIG_EXAMPLES_WEBSERVER_DHCPC
void *handle;
#endif
/* Many embedded network interfaces must have a software assigned MAC */
#ifdef CONFIG_EXAMPLES_WEBSERVER_NOMAC
mac[0] = 0x00;
mac[1] = 0xe0;
mac[2] = 0xde;
mac[3] = 0xad;
mac[4] = 0xbe;
mac[5] = 0xef;
netlib_setmacaddr("eth0", mac);
#endif
/* Set up our host address */
#ifdef CONFIG_EXAMPLES_WEBSERVER_DHCPC
addr.s_addr = 0;
#else
addr.s_addr = HTONL(CONFIG_EXAMPLES_WEBSERVER_IPADDR);
#endif
netlib_set_ipv4addr("eth0", &addr);
/* Set up the default router address */
addr.s_addr = HTONL(CONFIG_EXAMPLES_WEBSERVER_DRIPADDR);
netlib_set_dripv4addr("eth0", &addr);
/* Setup the subnet mask */
addr.s_addr = HTONL(CONFIG_EXAMPLES_WEBSERVER_NETMASK);
netlib_set_ipv4netmask("eth0", &addr);
2017-05-19 17:30:26 +02:00
/* New versions of netlib_set_ipvXaddr will not bring the network up,
2017-05-19 17:50:26 +02:00
* So ensure the network is really up at this point.
*/
2017-05-19 17:30:26 +02:00
netlib_ifup("eth0");
#ifdef CONFIG_EXAMPLES_WEBSERVER_DHCPC
/* Get the MAC address of the NIC */
netlib_getmacaddr("eth0", mac);
/* Set up the DHCPC modules */
handle = dhcpc_open("eth0", &mac, IFHWADDRLEN);
/* Get an IP address. Note: there is no logic here for renewing the
* address in this example. The address should be renewed in
* ds.lease_time/2 seconds.
*/
printf("Getting IP address\n");
if (handle)
{
struct dhcpc_state ds;
dhcpc_request(handle, &ds);
netlib_set_ipv4addr("eth0", &ds.ipaddr);
if (ds.netmask.s_addr != 0)
{
netlib_set_ipv4netmask("eth0", &ds.netmask);
}
if (ds.default_router.s_addr != 0)
{
netlib_set_dripv4addr("eth0", &ds.default_router);
}
if (ds.dnsaddr.s_addr != 0)
{
netlib_set_ipv4dnsaddr(&ds.dnsaddr);
}
dhcpc_close(handle);
printf("IP: %s\n", inet_ntoa(ds.ipaddr));
}
#endif
#endif /* CONFIG_NSH_NETINIT */
#ifdef CONFIG_NET_TCP
printf("Starting webserver\n");
httpd_init();
#ifndef CONFIG_NETUTILS_HTTPD_SCRIPT_DISABLE
cgi_register();
#endif
httpd_listen();
#endif
apps/examples/webserver: When NSH app, allow terminating The examples/webserver app can be built in two modes: (1) in standalone mode, or (2) as a NSH built-in app. When run in standalone mode, the webserver program is responsible for bringing up the network (including DHCP if configured). Also, the webserver program must never exit, so if httpd fails (i.e., if httpd_listen() returns), webserver_main() goes into an endless loop. When run as a NSH built-in app, network bring-up is the responsibility of other processes and the webserver program assumes the network is already properly configured when it starts. Also, if httpd_listen() returns, the webserver program should terminate. Prior to this change, the webserver program would *not* terminate, even when running as a NSH built-in app. For example: nsh> webserver & webserver [6:100] nsh> Starting webserver nsh> kill -9 6 nsh> webserver_main: Still running nsh> webserver_main: Still running nsh> webserver_main: Still running nsh> webserver_main: Still running The line "webserver_main: Still running" would be forever printed every 3 seconds, however httpd_listen() is no longer running and the webserver is not functional. This change makes the webserver play nicely when running as a NSH built-in app. With this change applied: nsh> webserver & webserver [6:100] nsh> Starting webserver nsh> kill -9 6 nsh> webserver_main: Exiting apps/examples/webserver/webserver_main.c: * main(): Infer from CONFIG_NSH_BUILTIN_APPS if this is a standalone program or a NSH built-in app. (See [1], where similar logic was added to decide whether to do network bring-up or not.) If standalone, run forever as before. If built-in app, exit when httpd terminates. References: [1] Commit 3a21b0b22263c06b5ba54ea2525521c8ca84b349
2021-01-03 23:58:04 +01:00
#ifndef CONFIG_NSH_NETINIT
/* We are running standalone (as opposed to a NSH built-in app). Therefore
* we should not exit after httpd failure.
*/
while (1)
{
sleep(3);
printf("webserver_main: Still running\n");
fflush(stdout);
}
apps/examples/webserver: When NSH app, allow terminating The examples/webserver app can be built in two modes: (1) in standalone mode, or (2) as a NSH built-in app. When run in standalone mode, the webserver program is responsible for bringing up the network (including DHCP if configured). Also, the webserver program must never exit, so if httpd fails (i.e., if httpd_listen() returns), webserver_main() goes into an endless loop. When run as a NSH built-in app, network bring-up is the responsibility of other processes and the webserver program assumes the network is already properly configured when it starts. Also, if httpd_listen() returns, the webserver program should terminate. Prior to this change, the webserver program would *not* terminate, even when running as a NSH built-in app. For example: nsh> webserver & webserver [6:100] nsh> Starting webserver nsh> kill -9 6 nsh> webserver_main: Still running nsh> webserver_main: Still running nsh> webserver_main: Still running nsh> webserver_main: Still running The line "webserver_main: Still running" would be forever printed every 3 seconds, however httpd_listen() is no longer running and the webserver is not functional. This change makes the webserver play nicely when running as a NSH built-in app. With this change applied: nsh> webserver & webserver [6:100] nsh> Starting webserver nsh> kill -9 6 nsh> webserver_main: Exiting apps/examples/webserver/webserver_main.c: * main(): Infer from CONFIG_NSH_BUILTIN_APPS if this is a standalone program or a NSH built-in app. (See [1], where similar logic was added to decide whether to do network bring-up or not.) If standalone, run forever as before. If built-in app, exit when httpd terminates. References: [1] Commit 3a21b0b22263c06b5ba54ea2525521c8ca84b349
2021-01-03 23:58:04 +01:00
#else /* CONFIG_NSH_NETINIT */
/* We are running as a NSH built-in app. Therefore we should exit. This
* allows to 'kill -9' the webserver app, assuming it was started as a
* background process. For example:
*
* nsh> webserver &
* webserver [6:100]
* nsh> Starting webserver
*
* nsh> kill -9 6
* nsh> webserver_main: Exiting
*/
printf("webserver_main: Exiting\n");
fflush(stdout);
#endif /* CONFIG_NSH_NETINIT */
return 0;
}