13c260e60Schristos /* 23c260e60Schristos * Hotspot 2.0 client - Web browser using wpadebug on Android 33c260e60Schristos * Copyright (c) 2013, Qualcomm Atheros, Inc. 43c260e60Schristos * 53c260e60Schristos * This software may be distributed under the terms of the BSD license. 63c260e60Schristos * See README for more details. 73c260e60Schristos */ 83c260e60Schristos 93c260e60Schristos #include "includes.h" 103c260e60Schristos 113c260e60Schristos #include "common.h" 123c260e60Schristos #include "utils/eloop.h" 133c260e60Schristos #include "wps/http_server.h" 143c260e60Schristos #include "browser.h" 153c260e60Schristos 163c260e60Schristos 173c260e60Schristos struct browser_data { 183c260e60Schristos int success; 193c260e60Schristos }; 203c260e60Schristos 213c260e60Schristos 223c260e60Schristos static void browser_timeout(void *eloop_data, void *user_ctx) 233c260e60Schristos { 243c260e60Schristos wpa_printf(MSG_INFO, "Timeout on waiting browser interaction to " 253c260e60Schristos "complete"); 263c260e60Schristos eloop_terminate(); 273c260e60Schristos } 283c260e60Schristos 293c260e60Schristos 303c260e60Schristos static void http_req(void *ctx, struct http_request *req) 313c260e60Schristos { 323c260e60Schristos struct browser_data *data = ctx; 333c260e60Schristos struct wpabuf *resp; 343c260e60Schristos const char *url; 353c260e60Schristos int done = 0; 363c260e60Schristos 373c260e60Schristos url = http_request_get_uri(req); 383c260e60Schristos wpa_printf(MSG_INFO, "Browser response received: %s", url); 393c260e60Schristos 403c260e60Schristos if (os_strcmp(url, "/") == 0) { 413c260e60Schristos data->success = 1; 423c260e60Schristos done = 1; 433c260e60Schristos } else if (os_strncmp(url, "/osu/", 5) == 0) { 443c260e60Schristos data->success = atoi(url + 5); 453c260e60Schristos done = 1; 463c260e60Schristos } 473c260e60Schristos 483c260e60Schristos resp = wpabuf_alloc(100); 493c260e60Schristos if (resp == NULL) { 503c260e60Schristos http_request_deinit(req); 513c260e60Schristos if (done) 523c260e60Schristos eloop_terminate(); 533c260e60Schristos return; 543c260e60Schristos } 550a73ee0aSchristos wpabuf_put_str(resp, "HTTP/1.1\r\n\r\nUser input completed"); 563c260e60Schristos 573c260e60Schristos if (done) { 583c260e60Schristos eloop_cancel_timeout(browser_timeout, NULL, NULL); 593c260e60Schristos eloop_register_timeout(0, 500000, browser_timeout, &data, NULL); 603c260e60Schristos } 613c260e60Schristos 623c260e60Schristos http_request_send_and_deinit(req, resp); 633c260e60Schristos } 643c260e60Schristos 653c260e60Schristos 66*bb618362Schristos int hs20_web_browser(const char *url, int ignore_tls) 673c260e60Schristos { 683c260e60Schristos struct http_server *http; 693c260e60Schristos struct in_addr addr; 703c260e60Schristos struct browser_data data; 71bb610346Schristos pid_t pid; 723c260e60Schristos 733c260e60Schristos wpa_printf(MSG_INFO, "Launching wpadebug browser to %s", url); 743c260e60Schristos 753c260e60Schristos os_memset(&data, 0, sizeof(data)); 763c260e60Schristos 773c260e60Schristos if (eloop_init() < 0) { 783c260e60Schristos wpa_printf(MSG_ERROR, "eloop_init failed"); 793c260e60Schristos return -1; 803c260e60Schristos } 813c260e60Schristos addr.s_addr = htonl((127 << 24) | 1); 823c260e60Schristos http = http_server_init(&addr, 12345, http_req, &data); 833c260e60Schristos if (http == NULL) { 843c260e60Schristos wpa_printf(MSG_ERROR, "http_server_init failed"); 853c260e60Schristos eloop_destroy(); 863c260e60Schristos return -1; 873c260e60Schristos } 883c260e60Schristos 89bb610346Schristos pid = fork(); 90bb610346Schristos if (pid < 0) { 91bb610346Schristos wpa_printf(MSG_ERROR, "fork: %s", strerror(errno)); 923c260e60Schristos http_server_deinit(http); 933c260e60Schristos eloop_destroy(); 943c260e60Schristos return -1; 953c260e60Schristos } 963c260e60Schristos 97bb610346Schristos if (pid == 0) { 98bb610346Schristos /* run the external command in the child process */ 9936ebd06eSchristos char *argv[14]; 1000a73ee0aSchristos char *envp[] = { "PATH=/system/bin:/vendor/bin", NULL }; 101bb610346Schristos 102bb610346Schristos argv[0] = "browser-wpadebug"; 103bb610346Schristos argv[1] = "start"; 104bb610346Schristos argv[2] = "-a"; 105bb610346Schristos argv[3] = "android.action.MAIN"; 106bb610346Schristos argv[4] = "-c"; 107bb610346Schristos argv[5] = "android.intent.category.LAUNCHER"; 108bb610346Schristos argv[6] = "-n"; 109bb610346Schristos argv[7] = "w1.fi.wpadebug/.WpaWebViewActivity"; 110bb610346Schristos argv[8] = "-e"; 111bb610346Schristos argv[9] = "w1.fi.wpadebug.URL"; 112bb610346Schristos argv[10] = (void *) url; 11336ebd06eSchristos argv[11] = "--user"; 11436ebd06eSchristos argv[12] = "-3"; /* USER_CURRENT_OR_SELF */ 11536ebd06eSchristos argv[13] = NULL; 116bb610346Schristos 1170a73ee0aSchristos execve("/system/bin/am", argv, envp); 1180a73ee0aSchristos wpa_printf(MSG_ERROR, "execve: %s", strerror(errno)); 119bb610346Schristos exit(0); 120bb610346Schristos return -1; 121bb610346Schristos } 122bb610346Schristos 1233c260e60Schristos eloop_register_timeout(300, 0, browser_timeout, &data, NULL); 1243c260e60Schristos eloop_run(); 1253c260e60Schristos eloop_cancel_timeout(browser_timeout, &data, NULL); 1263c260e60Schristos http_server_deinit(http); 1273c260e60Schristos eloop_destroy(); 1283c260e60Schristos 1293c260e60Schristos wpa_printf(MSG_INFO, "Closing Android browser"); 1303c260e60Schristos if (os_exec("/system/bin/am", 1313c260e60Schristos "start -a android.action.MAIN " 1323c260e60Schristos "-c android.intent.category.LAUNCHER " 1333c260e60Schristos "-n w1.fi.wpadebug/.WpaWebViewActivity " 1343c260e60Schristos "-e w1.fi.wpadebug.URL FINISH", 1) != 0) { 1353c260e60Schristos wpa_printf(MSG_INFO, "Failed to close wpadebug browser"); 1363c260e60Schristos } 1373c260e60Schristos 1383c260e60Schristos return data.success; 1393c260e60Schristos } 140