The patch titled ecryptfs: Validate packet length prior to parsing, add comments has been added to the -mm tree. Its filename is ecryptfs-validate-packet-length-prior-to-parsing-add-comments.patch See http://www.zip.com.au/~akpm/linux/patches/stuff/added-to-mm.txt to find out what to do about this ------------------------------------------------------ Subject: ecryptfs: Validate packet length prior to parsing, add comments From: Stephan Mueller <smueller@xxxxxxxxxx> The parsing of the encrypted file's meta data page is done byte by byte. The current implementation checks after each read byte whether it is still in the allowed range to be read. The patch changes that in the sense that: - the check is done before the actual byte is read - the number of checks is reduced as the format that we read is fix to make the code more readable and better maintainable Two blocks are expected in the meta data page: Tag 3 and Tag 11. The following formats are being considered: Tag 3: one byte for the Tag 3 ID flag two bytes for the body size five fix bytes for: version string, cipher, S2K ID, hash algo, number of hash iterations ECRYPTFS_SALT_SIZE bytes for salt body_size bytes minus ECRYPTFS_SALT_SIZE minus 5 is the encrypted key size Tag 11: one byte for the Tag 11 ID flag two bytes for the Tag 11 length 14 bytes for: special flag one, special flag two, 12 skipped bytes body_size bytes minus 14 bytes is the Tag 11 content In addition, the patch: - makes some branches unlikely() - implicitly fixes two memory leaks as goto out instead of goto out_free were was used Signed-off-by: Stephan Mueller <smueller@xxxxxxxxxx> Acked-by: Michael Halcrow <mhalcrow@xxxxxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxx> --- fs/ecryptfs/keystore.c | 160 ++++++++++++++++++--------------------- 1 files changed, 77 insertions(+), 83 deletions(-) diff -puN fs/ecryptfs/keystore.c~ecryptfs-validate-packet-length-prior-to-parsing-add-comments fs/ecryptfs/keystore.c --- 25/fs/ecryptfs/keystore.c~ecryptfs-validate-packet-length-prior-to-parsing-add-comments Tue Jun 27 15:09:35 2006 +++ 25-akpm/fs/ecryptfs/keystore.c Tue Jun 27 15:09:35 2006 @@ -180,19 +180,27 @@ parse_tag_3_packet(struct ecryptfs_crypt struct ecryptfs_auth_tok_list_item *auth_tok_list_item; int length_size; + /* we check that: + * one byte for the Tag 3 ID flag + * two bytes for the body size + * do not exceed the maximum_packet_size + */ + if (unlikely((*packet_size) + 3 > max_packet_size)) { + ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); + rc = -EINVAL; + goto out; + } + (*packet_size) = 0; (*new_auth_tok) = NULL; + + /* check for Tag 3 identifyer - one byte */ if (data[(*packet_size)++] != ECRYPTFS_TAG_3_PACKET_TYPE) { ecryptfs_printk(KERN_ERR, "Enter w/ first byte != 0x%.2x\n", ECRYPTFS_TAG_3_PACKET_TYPE); rc = -EINVAL; goto out; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } /* Released: wipe_auth_tok_list called in ecryptfs_parse_packet_set or * at end of function upon failure */ auth_tok_list_item = @@ -205,6 +213,8 @@ parse_tag_3_packet(struct ecryptfs_crypt memset(auth_tok_list_item, 0, sizeof(struct ecryptfs_auth_tok_list_item)); (*new_auth_tok) = &auth_tok_list_item->auth_tok; + + /* check for body size - one to two bytes */ rc = parse_packet_length(&data[(*packet_size)], &body_size, &length_size); if (rc) { @@ -212,36 +222,42 @@ parse_tag_3_packet(struct ecryptfs_crypt "rc = [%d]\n", rc); goto out_free; } - if (body_size < (0x05 + ECRYPTFS_SALT_SIZE)) { + if (unlikely(body_size < (0x05 + ECRYPTFS_SALT_SIZE))) { ecryptfs_printk(KERN_WARNING, "Invalid body size ([%d])\n", body_size); rc = -EINVAL; goto out_free; } (*packet_size) += length_size; - if (unlikely((*packet_size) > max_packet_size)) { + + /* now we know the length of the remainting Tag 3 packet size: + * 5 fix bytes for: version string, cipher, S2K ID, hash algo, + * number of hash iterations + * ECRYPTFS_SALT_SIZE bytes for salt + * body_size bytes minus the stuff above is the encrypted key size + */ + if (unlikely((*packet_size) + body_size > max_packet_size)) { ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); rc = -EINVAL; goto out_free; } + /* There are 5 characters of additional information in the * packet */ (*new_auth_tok)->session_key.encrypted_key_size = body_size - (0x05 + ECRYPTFS_SALT_SIZE); ecryptfs_printk(KERN_DEBUG, "Encrypted key size = [%d]\n", (*new_auth_tok)->session_key.encrypted_key_size); - /* Version 4 (from RFC2440) */ - if (data[(*packet_size)++] != 0x04) { + + /* Version 4 (from RFC2440) - one byte */ + if (unlikely(data[(*packet_size)++] != 0x04)) { ecryptfs_printk(KERN_DEBUG, "Unknown version number " "[%d]\n", data[(*packet_size) - 1]); rc = -EINVAL; goto out_free; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } + + /* cipher - one byte */ ecryptfs_cipher_code_to_string(crypt_stat->cipher, (u16)data[(*packet_size)]); /* A little extra work to differentiate among the AES key @@ -254,59 +270,39 @@ parse_tag_3_packet(struct ecryptfs_crypt crypt_stat->key_size = (*new_auth_tok)->session_key.encrypted_key_size; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out_free; - } ecryptfs_init_crypt_ctx(crypt_stat); /* S2K identifier 3 (from RFC2440) */ - if (data[(*packet_size)++] != 0x03) { + if (unlikely(data[(*packet_size)++] != 0x03)) { ecryptfs_printk(KERN_ERR, "Only S2K ID 3 is currently " "supported\n"); rc = -ENOSYS; goto out_free; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out_free; - } + /* TODO: finish the hash mapping */ + /* hash algorithm - one byte */ switch (data[(*packet_size)++]) { case 0x01: /* See RFC2440 for these numbers and their mappings */ /* Choose MD5 */ + /* salt - ECRYPTFS_SALT_SIZE bytes */ memcpy((*new_auth_tok)->token.password.salt, &data[(*packet_size)], ECRYPTFS_SALT_SIZE); (*packet_size) += ECRYPTFS_SALT_SIZE; - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, - "Packet size exceeds max\n"); - rc = -EINVAL; - goto out_free; - } + /* This conversion was taken straight from RFC2440 */ + /* number of hash iterations - one byte */ (*new_auth_tok)->token.password.hash_iterations = ((u32) 16 + (data[(*packet_size)] & 15)) << ((data[(*packet_size)] >> 4) + 6); (*packet_size)++; - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, - "Packet size exceeds max\n"); - rc = -EINVAL; - goto out_free; - } + + /* encrypted session key - + * (body_size-5-ECRYPTFS_SALT_SIZE) bytes */ memcpy((*new_auth_tok)->session_key.encrypted_key, &data[(*packet_size)], (*new_auth_tok)->session_key.encrypted_key_size); (*packet_size) += (*new_auth_tok)->session_key.encrypted_key_size; - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, - "Packet size exceeds max\n"); - rc = -EINVAL; - goto out_free; - } (*new_auth_tok)->session_key.flags &= ~ECRYPTFS_CONTAINS_DECRYPTED_KEY; (*new_auth_tok)->session_key.flags |= @@ -319,11 +315,6 @@ parse_tag_3_packet(struct ecryptfs_crypt rc = -ENOSYS; goto out_free; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } /* TODO: Use the keyring */ (*new_auth_tok)->uid = current->uid; ECRYPTFS_SET_FLAG((*new_auth_tok)->flags, ECRYPTFS_PASSWORD); @@ -373,17 +364,27 @@ parse_tag_11_packet(unsigned char *data, (*packet_size) = 0; (*tag_11_contents_size) = 0; - if (data[(*packet_size)++] != ECRYPTFS_TAG_11_PACKET_TYPE) { - ecryptfs_printk(KERN_WARNING, - "Invalid tag 11 packet format\n"); + + /* check that: + * one byte for the Tag 11 ID flag + * two bytes for the Tag 11 length + * do not exceed the maximum_packet_size + */ + if (unlikely((*packet_size) + 3 > max_packet_size)) { + ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); rc = -EINVAL; goto out; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); + + /* check for Tag 11 identifyer - one byte */ + if (data[(*packet_size)++] != ECRYPTFS_TAG_11_PACKET_TYPE) { + ecryptfs_printk(KERN_WARNING, + "Invalid tag 11 packet format\n"); rc = -EINVAL; goto out; } + + /* get Tag 11 content length - one or two bytes */ rc = parse_packet_length(&data[(*packet_size)], &body_size, &length_size); if (rc) { @@ -392,11 +393,7 @@ parse_tag_11_packet(unsigned char *data, goto out; } (*packet_size) += length_size; - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } + if (body_size < 13) { ecryptfs_printk(KERN_WARNING, "Invalid body size ([%d])\n", body_size); @@ -405,47 +402,44 @@ parse_tag_11_packet(unsigned char *data, } /* We have 13 bytes of surrounding packet values */ (*tag_11_contents_size) = (body_size - 13); - if ((*tag_11_contents_size) > max_contents_bytes) { - rc = -ENOMEM; - ecryptfs_printk(KERN_WARNING, "Not enough space allocated " - "in contents to copy entire contents of tag 11 " - "packet\n"); + + /* now we know the length of the remainting Tag 11 packet size: + * 14 fix bytes for: special flag one, special flag two, + * 12 skipped bytes + * body_size bytes minus the stuff above is the Tag 11 content + */ + /* FIXME why is the body size one byte smaller than the actual + * size of the body? + * this seems to be an error here as well as in + * write_tag_11_packet() */ + if (unlikely((*packet_size) + body_size + 1 > max_packet_size)) { + ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); + rc = -EINVAL; goto out; } + + /* special flag one - one byte */ if (data[(*packet_size)++] != 0x62) { ecryptfs_printk(KERN_WARNING, "Unrecognizable packet\n"); rc = -EINVAL; goto out; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } + + /* special flag two - one byte */ if (data[(*packet_size)++] != 0x08) { ecryptfs_printk(KERN_WARNING, "Unrecognizable packet\n"); rc = -EINVAL; goto out; } - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } + + /* skip the next 12 bytes */ (*packet_size) += 12; /* We don't care about the filename or * the timestamp */ - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } + + /* get the Tag 11 contents - tag_11_contents_size bytes */ memcpy(contents, &data[(*packet_size)], (*tag_11_contents_size)); (*packet_size) += (*tag_11_contents_size); - if (unlikely((*packet_size) > max_packet_size)) { - ecryptfs_printk(KERN_ERR, "Packet size exceeds max\n"); - rc = -EINVAL; - goto out; - } + out: if (rc) { (*packet_size) = 0; _ Patches currently in -mm which might be from smueller@xxxxxxxxxx are ecryptfs-validate-minimum-header-extent-size.patch ecryptfs-validate-body-size.patch ecryptfs-validate-packet-length-prior-to-parsing-add-comments.patch ecryptfs-validate-packet-length-prior-to-parsing-add-comments-fix.patch ecryptfs-use-the-passed-in-max-value-as-the-upper-bound.patch ecryptfs-change-the-maximum-size-check-when-writing-header.patch ecryptfs-print-the-actual-option-that-is-problematic.patch - To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html