139 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			139 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			C
		
	
	
	
/* Copyright 2021 Google LLC
 | 
						|
Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
you may not use this file except in compliance with the License.
 | 
						|
You may obtain a copy of the License at
 | 
						|
      http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
Unless required by applicable law or agreed to in writing, software
 | 
						|
distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
See the License for the specific language governing permissions and
 | 
						|
limitations under the License.
 | 
						|
*/
 | 
						|
#include "apr.h"
 | 
						|
#include "apr_file_io.h"
 | 
						|
#include "apr_poll.h"
 | 
						|
#include "apr_portable.h"
 | 
						|
#include "apr_proc_mutex.h"
 | 
						|
#include "apr_signal.h"
 | 
						|
#include "apr_strings.h"
 | 
						|
#include "apr_thread_mutex.h"
 | 
						|
#include "apr_thread_proc.h"
 | 
						|
#include "http_core.h"
 | 
						|
 | 
						|
#define APR_WANT_STRFUNC
 | 
						|
#include "apr_file_io.h"
 | 
						|
#include "apr_fnmatch.h"
 | 
						|
#include "apr_want.h"
 | 
						|
 | 
						|
#include "apr_poll.h"
 | 
						|
#include "apr_want.h"
 | 
						|
 | 
						|
#include "ap_config.h"
 | 
						|
#include "ap_expr.h"
 | 
						|
#include "ap_listen.h"
 | 
						|
#include "ap_provider.h"
 | 
						|
#include "ap_regex.h"
 | 
						|
 | 
						|
#include "ada_fuzz_header.h"
 | 
						|
 | 
						|
static const char *http_scheme2(const request_rec *r) {
 | 
						|
  /*
 | 
						|
   * The http module shouldn't return anything other than
 | 
						|
   * "http" (the default) or "https".
 | 
						|
   */
 | 
						|
  if (r->server->server_scheme &&
 | 
						|
      (strcmp(r->server->server_scheme, "https") == 0))
 | 
						|
    return "https";
 | 
						|
 | 
						|
  return "http";
 | 
						|
}
 | 
						|
 | 
						|
extern request_rec *ap_create_request(conn_rec *conn);
 | 
						|
extern int read_request_line(request_rec *r, apr_bucket_brigade *bb);
 | 
						|
 | 
						|
int LLVMFuzzerInitialize(int *argc, char ***argv) {
 | 
						|
  apr_pool_create(&apr_hook_global_pool, NULL);
 | 
						|
  ap_open_stderr_log(apr_hook_global_pool);
 | 
						|
  ap_hook_http_scheme(http_scheme2, NULL, NULL, APR_HOOK_REALLY_LAST);
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
 | 
						|
  af_gb_init();
 | 
						|
 | 
						|
  const uint8_t *data2 = data;
 | 
						|
  size_t size2 = size;
 | 
						|
 | 
						|
  /* get random data for the fuzzer */
 | 
						|
  char *new_str = af_gb_get_null_terminated(&data2, &size2);
 | 
						|
  char *new_str2 = af_gb_get_null_terminated(&data2, &size2);
 | 
						|
  char *new_str3 = af_gb_get_null_terminated(&data2, &size2);
 | 
						|
  char *new_str4 = af_gb_get_null_terminated(&data2, &size2);
 | 
						|
  char *new_str5 = af_gb_get_null_terminated(&data2, &size2);
 | 
						|
  if (new_str != NULL && 
 | 
						|
      new_str2 != NULL && 
 | 
						|
      new_str3 != NULL &&
 | 
						|
      new_str4 != NULL && 
 | 
						|
      new_str5 != NULL) {
 | 
						|
 | 
						|
    /* this is the main fuzzing logic */
 | 
						|
 | 
						|
    apr_pool_initialize();
 | 
						|
    apr_pool_t *v = NULL;
 | 
						|
    apr_pool_create(&v, NULL);
 | 
						|
 | 
						|
    conn_rec conn;
 | 
						|
    conn.pool = v;
 | 
						|
    server_rec base_server;
 | 
						|
    conn.base_server = &base_server;
 | 
						|
    conn.bucket_alloc = apr_bucket_alloc_create(conn.pool);
 | 
						|
    ap_method_registry_init(conn.pool);
 | 
						|
 | 
						|
    //server_rec server;
 | 
						|
 | 
						|
    /* Simulate ap_read_request */
 | 
						|
    request_rec *r = NULL;
 | 
						|
    r = ap_create_request(&conn);
 | 
						|
 | 
						|
    /* create a logs array for the request */
 | 
						|
    struct ap_logconf logs = {};
 | 
						|
    char *log_levels = calloc(1000, 1);
 | 
						|
    memset(log_levels, 0, 1000);
 | 
						|
    logs.module_levels = log_levels;
 | 
						|
    r->log = &logs;
 | 
						|
    if (r != NULL) {
 | 
						|
      apr_bucket_brigade *tmp_bb = apr_brigade_create(r->pool, r->connection->bucket_alloc);
 | 
						|
      conn.keepalive = AP_CONN_UNKNOWN;
 | 
						|
 | 
						|
      ap_run_pre_read_request(r, conn);
 | 
						|
 | 
						|
      core_server_config conf_mod;
 | 
						|
      conf_mod.http_conformance   = (char)af_get_short(&data2, &size2);
 | 
						|
      conf_mod.http09_enable      = (char)af_get_short(&data2, &size2);
 | 
						|
      conf_mod.http_methods       = (char)af_get_short(&data2, &size2);
 | 
						|
      void **module_config_arr = malloc(1000);
 | 
						|
      module_config_arr[0] = &conf_mod;
 | 
						|
 | 
						|
      r->server->module_config = module_config_arr;
 | 
						|
      ap_set_core_module_config(r->server->module_config, &conf_mod);
 | 
						|
 | 
						|
      /* randomise content of request */
 | 
						|
      r->unparsed_uri           = new_str;
 | 
						|
      r->uri                    = new_str2;
 | 
						|
      r->server->server_scheme  = new_str3;
 | 
						|
      r->method                 = new_str4;
 | 
						|
      r->the_request            = new_str5;
 | 
						|
 | 
						|
      /* main target */
 | 
						|
      ap_parse_request_line(r);
 | 
						|
 | 
						|
      free(module_config_arr);
 | 
						|
    }
 | 
						|
    free(log_levels);
 | 
						|
    apr_pool_terminate();
 | 
						|
  }
 | 
						|
 | 
						|
  af_gb_cleanup();
 | 
						|
  return 0;
 | 
						|
}
 |