diff --git a/src/hci.c b/src/hci.c
index 99f0ed4d9..eeba80161 100644
--- a/src/hci.c
+++ b/src/hci.c
@@ -5338,12 +5338,17 @@ void hci_setup_test_connections_fuzz(void){
     // default address: 66:55:44:33:00:01
     bd_addr_t addr = { 0x66, 0x55, 0x44, 0x33, 0x00, 0x00};
 
+    // setup Controller info
+    hci_stack->num_cmd_packets = 255;
+    hci_stack->acl_packets_total_num = 255;
+
     // setup incoming Classic ACL connection with con handle 0x0001, 66:55:44:33:22:01
     addr[5] = 0x01;
     conn = create_connection_for_bd_addr_and_type(addr, BD_ADDR_TYPE_ACL);
     conn->con_handle = addr[5];
     conn->role  = HCI_ROLE_SLAVE;
     conn->state = RECEIVED_CONNECTION_REQUEST;
+    conn->sm_connection.sm_role = HCI_ROLE_SLAVE;
 
     // setup incoming Classic SCO connection with con handle 0x0002
     addr[5] = 0x02;
@@ -5351,6 +5356,7 @@ void hci_setup_test_connections_fuzz(void){
     conn->con_handle = addr[5];
     conn->role  = HCI_ROLE_SLAVE;
     conn->state = RECEIVED_CONNECTION_REQUEST;
+    conn->sm_connection.sm_role = HCI_ROLE_SLAVE;
 
     // setup ready Classic ACL connection with con handle 0x0003
     addr[5] = 0x03;
@@ -5358,6 +5364,7 @@ void hci_setup_test_connections_fuzz(void){
     conn->con_handle = addr[5];
     conn->role  = HCI_ROLE_SLAVE;
     conn->state = OPEN;
+    conn->sm_connection.sm_role = HCI_ROLE_SLAVE;
 
     // setup ready Classic SCO connection with con handle 0x0004
     addr[5] = 0x04;
@@ -5365,6 +5372,7 @@ void hci_setup_test_connections_fuzz(void){
     conn->con_handle = addr[5];
     conn->role  = HCI_ROLE_SLAVE;
     conn->state = OPEN;
+    conn->sm_connection.sm_role = HCI_ROLE_SLAVE;
 
     // setup ready LE ACL connection with con handle 0x005 and public address
     addr[5] = 0x05;
@@ -5372,6 +5380,7 @@ void hci_setup_test_connections_fuzz(void){
     conn->con_handle = addr[5];
     conn->role  = HCI_ROLE_SLAVE;
     conn->state = OPEN;
+    conn->sm_connection.sm_role = HCI_ROLE_SLAVE;
 }
 
 void hci_free_connections_fuzz(void){
diff --git a/test/fuzz/fuzz_gatt_client.c b/test/fuzz/fuzz_gatt_client.c
index b59b6d8ac..9a5ce420c 100644
--- a/test/fuzz/fuzz_gatt_client.c
+++ b/test/fuzz/fuzz_gatt_client.c
@@ -50,8 +50,37 @@ static const hci_transport_t hci_transport_fuzz = {
 static void gatt_client_packet_handler(uint8_t packet_type, uint16_t handle, uint8_t *packet, uint16_t size){
 }
 
-int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
+void set_gatt_service_uuid16(gatt_client_service_t * service, const uint8_t *data, size_t size){
+    service->start_group_handle = 0x0001;
+    service->end_group_handle = 0xffff;
+    memset(service->uuid128, 0, 16);
+    service->uuid16 = big_endian_read_16(data, 0);
+}
 
+void set_gatt_service_uuid128(gatt_client_service_t * service, const uint8_t *data, size_t size){
+    service->start_group_handle = 0x0001;
+    service->end_group_handle = 0xffff;
+    service->uuid16 = 0;
+    memcpy(service->uuid128, data, 16);
+}
+
+void set_gatt_characteristic_uuid16(gatt_client_characteristic_t * characteristic, const uint8_t *data, size_t size){
+    characteristic->start_handle = 0x0001;
+    characteristic->value_handle = 0x0002;
+    characteristic->end_handle = 0xffff;
+    characteristic->uuid16 = big_endian_read_16(data, 0);
+    memset(characteristic->uuid128, 0, 16);
+}
+
+void set_gatt_characteristic_uuid128(gatt_client_characteristic_t * characteristic, const uint8_t *data, size_t size){
+    characteristic->start_handle = 0x0001;
+    characteristic->value_handle = 0x0002;
+    characteristic->end_handle = 0xffff;
+    characteristic->uuid16 = 0;
+    memcpy(characteristic->uuid128, data, 16);
+}
+
+int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
     const hci_con_handle_t ble_handle = 0x0005;
 
     static bool gatt_client_initiated = false;
@@ -63,13 +92,163 @@ int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
         hci_setup_test_connections_fuzz();
 
         gatt_client_init();
+        gatt_client_mtu_enable_auto_negotiation(0);
         gatt_client_initiated = true;
     }
 
-    // TODO: use first byte of random data to pick gatt_client request / set gatt client state
-    // then, only use dat from second byte as response
-    gatt_client_discover_primary_services(gatt_client_packet_handler, ble_handle);
+    // use first byte of random data to pick gatt_client request / set gatt client state
+    // then, only use data from second byte as response
+    // prepare test data
+    if (size < 1) return 0;
+    uint8_t  cmd_type  = (data[0] & 0x1F); 
+    size--;
+    data++;
 
+    uint8_t  uuid128[16];
+    uint16_t uuid16;
+    int offset = 0;
+    gatt_client_service_t service;
+    gatt_client_characteristic_t characteristic;
+    gatt_client_characteristic_descriptor_t descriptor;
+
+    switch (cmd_type){
+        case 1:
+            gatt_client_discover_primary_services(gatt_client_packet_handler, ble_handle);
+            break;
+        case 2:
+            offset = 2;
+            if (size < offset) return 0;
+            uuid16 = big_endian_read_16(data, 0);
+            gatt_client_discover_primary_services_by_uuid16(gatt_client_packet_handler, ble_handle, uuid16);
+            break;
+        case 3:
+            offset = 16;
+            if (size < offset) return 0;
+            memcpy(uuid128, data, 16);
+            gatt_client_discover_primary_services_by_uuid128(gatt_client_packet_handler, ble_handle, uuid128);
+            break;
+        case 4:
+            offset = 2;
+            if (size < offset) return 0;
+            set_gatt_service_uuid16(&service, data, size);
+            gatt_client_find_included_services_for_service(gatt_client_packet_handler, ble_handle, &service);
+            break;
+        case 5:
+            offset = 2;
+            if (size < offset) return 0;
+            set_gatt_service_uuid16(&service, data, size);
+            gatt_client_discover_characteristics_for_service(gatt_client_packet_handler, ble_handle, &service);
+            break;
+        case 6:
+            offset = 2;
+            if (size < offset) return 0;
+            uuid16 = big_endian_read_16(data, 0);
+            gatt_client_discover_characteristics_for_handle_range_by_uuid16(gatt_client_packet_handler, ble_handle, 0x0001, 0xffff, uuid16);
+            break;
+        case 7:
+            offset = 16;
+            if (size < offset) return 0;
+            memcpy(uuid128, data, 16);
+            gatt_client_discover_characteristics_for_handle_range_by_uuid128(gatt_client_packet_handler, ble_handle, 0x0001, 0xffff, uuid128);
+            break;
+        case 8:
+            offset = 4;
+            if (size < offset) return 0;
+            set_gatt_service_uuid16(&service, data, size);
+            uuid16 = big_endian_read_16(data, 2);
+            gatt_client_discover_characteristics_for_service_by_uuid16(gatt_client_packet_handler, ble_handle, &service, uuid16);
+            break;
+        case 9:
+            offset = 18;
+            if (size < offset) return 0;
+            set_gatt_service_uuid16(&service, data, size);
+            memcpy(uuid128, data + 2, 16);
+            gatt_client_discover_characteristics_for_service_by_uuid128(gatt_client_packet_handler, ble_handle, &service, uuid128);
+            break;
+        case 10:
+            offset = 2;
+            if (size < offset) return 0;
+            set_gatt_characteristic_uuid16(&characteristic, data, size);
+            gatt_client_discover_characteristic_descriptors(gatt_client_packet_handler, ble_handle, &characteristic);
+            break;
+        case 11:
+            offset = 2;
+            if (size < offset) return 0;
+            set_gatt_characteristic_uuid16(&characteristic, data, size);
+            gatt_client_read_value_of_characteristic(gatt_client_packet_handler, ble_handle, &characteristic);
+            break;
+        case 12:
+            offset = 2;
+            if (size < offset) return 0;
+            set_gatt_characteristic_uuid16(&characteristic, data, size);
+            gatt_client_read_value_of_characteristics_by_uuid16(gatt_client_packet_handler, ble_handle, characteristic.start_handle, characteristic.end_handle, characteristic.uuid16);
+            break;
+        case 13:
+            offset = 16;
+            if (size < offset) return 0;
+            set_gatt_characteristic_uuid128(&characteristic, data, size);
+            gatt_client_read_value_of_characteristics_by_uuid128(gatt_client_packet_handler, ble_handle, characteristic.start_handle, characteristic.end_handle, characteristic.uuid128);
+            break;
+        case 14:
+            offset = 2;
+            if (size < offset) return 0;
+            set_gatt_characteristic_uuid16(&characteristic, data, size);
+            gatt_client_read_long_value_of_characteristic(gatt_client_packet_handler, ble_handle, &characteristic);
+            break;
+        case 15:
+            offset = 4;
+            if (size < offset) return 0;
+            set_gatt_characteristic_uuid16(&characteristic, data, size);
+            gatt_client_read_long_value_of_characteristic_using_value_handle_with_offset(gatt_client_packet_handler, ble_handle, characteristic.value_handle, big_endian_read_16(data, 2));
+            break;
+        case 16: 
+            gatt_client_read_multiple_characteristic_values(gatt_client_packet_handler, ble_handle, 0, NULL);
+            break;
+        case 17:
+            gatt_client_write_value_of_characteristic(gatt_client_packet_handler, ble_handle, 5, 0, NULL);
+            break;
+        case 18:
+            gatt_client_write_long_value_of_characteristic(gatt_client_packet_handler, ble_handle, 5, 0, NULL);
+            break;
+        case 19:
+            gatt_client_reliable_write_long_value_of_characteristic(gatt_client_packet_handler, ble_handle, 5, 0, NULL);
+            break;
+        case 20:
+            gatt_client_read_characteristic_descriptor_using_descriptor_handle(gatt_client_packet_handler, ble_handle, 5);
+            break;
+        case 21:
+            gatt_client_read_long_characteristic_descriptor_using_descriptor_handle(gatt_client_packet_handler, ble_handle, 5);
+            break;
+        case 22:
+            gatt_client_write_characteristic_descriptor_using_descriptor_handle(gatt_client_packet_handler, ble_handle, 5, 0, NULL);
+            break;
+        case 23:
+            gatt_client_write_long_characteristic_descriptor_using_descriptor_handle(gatt_client_packet_handler, ble_handle, 5, 0, NULL);
+            break;
+        case 24:
+            offset = 2;
+            if (size < offset) return 0;
+            set_gatt_characteristic_uuid16(&characteristic, data, size);
+            gatt_client_write_client_characteristic_configuration(gatt_client_packet_handler, ble_handle, &characteristic, 1);
+            break;
+        case 25:
+            gatt_client_prepare_write(gatt_client_packet_handler, ble_handle, 5, 0, 0, NULL);
+            break;
+        case 26:
+            gatt_client_prepare_write(gatt_client_packet_handler, ble_handle, 5, 0, 0, NULL);
+            gatt_client_execute_write(gatt_client_packet_handler, ble_handle);
+            break;
+        case 27:
+            gatt_client_prepare_write(gatt_client_packet_handler, ble_handle, 5, 0, 0, NULL);
+            gatt_client_cancel_write(gatt_client_packet_handler, ble_handle);
+            break;
+        default:
+            return 0;
+    }
+
+    data += offset;
+    size -= offset;
+    
     // send test response
     gatt_client_att_packet_handler_fuzz(ATT_DATA_PACKET, ble_handle, (uint8_t *) data, size);
     return 0;