handle);
                conn->enc_key_size = 0;
        } else {
+               u8 *key_enc_size = hci_conn_key_enc_size(conn);
+
                conn->enc_key_size = rp->key_size;
                status = 0;
 
-               if (conn->enc_key_size < hdev->min_enc_key_size) {
+               /* Attempt to check if the key size is too small or if it has
+                * been downgraded from the last time it was stored as part of
+                * the link_key.
+                */
+               if (conn->enc_key_size < hdev->min_enc_key_size ||
+                   (key_enc_size && conn->enc_key_size < *key_enc_size)) {
                        /* As slave role, the conn->state has been set to
                         * BT_CONNECTED and l2cap conn req might not be received
                         * yet, at this moment the l2cap layer almost does
                        clear_bit(HCI_CONN_ENCRYPT, &conn->flags);
                        clear_bit(HCI_CONN_AES_CCM, &conn->flags);
                }
+
+               /* Update the key encryption size with the connection one */
+               if (key_enc_size && *key_enc_size != conn->enc_key_size)
+                       *key_enc_size = conn->enc_key_size;
        }
 
        hci_encrypt_cfm(conn, status);
        hci_dev_unlock(hdev);
 }
 
+static int hci_read_enc_key_size(struct hci_dev *hdev, struct hci_conn *conn)
+{
+       struct hci_cp_read_enc_key_size cp;
+       u8 *key_enc_size = hci_conn_key_enc_size(conn);
+
+       if (!read_key_size_capable(hdev)) {
+               conn->enc_key_size = HCI_LINK_KEY_SIZE;
+               return -EOPNOTSUPP;
+       }
+
+       bt_dev_dbg(hdev, "hcon %p", conn);
+
+       memset(&cp, 0, sizeof(cp));
+       cp.handle = cpu_to_le16(conn->handle);
+
+       /* If the key enc_size is already known, use it as conn->enc_key_size,
+        * otherwise use hdev->min_enc_key_size so the likes of
+        * l2cap_check_enc_key_size don't fail while waiting for
+        * HCI_OP_READ_ENC_KEY_SIZE response.
+        */
+       if (key_enc_size && *key_enc_size)
+               conn->enc_key_size = *key_enc_size;
+       else
+               conn->enc_key_size = hdev->min_enc_key_size;
+
+       return hci_send_cmd(hdev, HCI_OP_READ_ENC_KEY_SIZE, sizeof(cp), &cp);
+}
+
 static void hci_conn_complete_evt(struct hci_dev *hdev, void *data,
                                  struct sk_buff *skb)
 {
                if (ev->encr_mode == 1 && !test_bit(HCI_CONN_ENCRYPT, &conn->flags) &&
                    ev->link_type == ACL_LINK) {
                        struct link_key *key;
-                       struct hci_cp_read_enc_key_size cp;
 
                        key = hci_find_link_key(hdev, &ev->bdaddr);
                        if (key) {
                                set_bit(HCI_CONN_ENCRYPT, &conn->flags);
-
-                               if (!read_key_size_capable(hdev)) {
-                                       conn->enc_key_size = HCI_LINK_KEY_SIZE;
-                               } else {
-                                       cp.handle = cpu_to_le16(conn->handle);
-                                       if (hci_send_cmd(hdev, HCI_OP_READ_ENC_KEY_SIZE,
-                                                        sizeof(cp), &cp)) {
-                                               bt_dev_err(hdev, "sending read key size failed");
-                                               conn->enc_key_size = HCI_LINK_KEY_SIZE;
-                                       }
-                               }
-
+                               hci_read_enc_key_size(hdev, conn);
                                hci_encrypt_cfm(conn, ev->status);
                        }
                }
 
        /* Try reading the encryption key size for encrypted ACL links */
        if (!ev->status && ev->encrypt && conn->type == ACL_LINK) {
-               struct hci_cp_read_enc_key_size cp;
-
-               /* Only send HCI_Read_Encryption_Key_Size if the
-                * controller really supports it. If it doesn't, assume
-                * the default size (16).
-                */
-               if (!read_key_size_capable(hdev)) {
-                       conn->enc_key_size = HCI_LINK_KEY_SIZE;
-                       goto notify;
-               }
-
-               cp.handle = cpu_to_le16(conn->handle);
-               if (hci_send_cmd(hdev, HCI_OP_READ_ENC_KEY_SIZE,
-                                sizeof(cp), &cp)) {
-                       bt_dev_err(hdev, "sending read key size failed");
-                       conn->enc_key_size = HCI_LINK_KEY_SIZE;
+               if (hci_read_enc_key_size(hdev, conn))
                        goto notify;
-               }
 
                goto unlock;
        }