diff options
Diffstat (limited to 'erts/emulator/beam/erl_bits.c')
| -rw-r--r-- | erts/emulator/beam/erl_bits.c | 54 | 
1 files changed, 31 insertions, 23 deletions
| diff --git a/erts/emulator/beam/erl_bits.c b/erts/emulator/beam/erl_bits.c index 6bf52fb303..51d23a8965 100644 --- a/erts/emulator/beam/erl_bits.c +++ b/erts/emulator/beam/erl_bits.c @@ -1,7 +1,7 @@  /*   * %CopyrightBegin%   * - * Copyright Ericsson AB 1999-2016. All Rights Reserved. + * Copyright Ericsson AB 1999-2017. All Rights Reserved.   *   * Licensed under the Apache License, Version 2.0 (the "License");   * you may not use this file except in compliance with the License. @@ -32,15 +32,6 @@  #include "erl_bits.h"  #include "erl_binary.h" -#ifdef MAX -#undef MAX -#endif -#define MAX(x,y) (((x)>(y))?(x):(y)) -#ifdef MIN -#undef MIN -#endif -#define MIN(x,y) (((x)<(y))?(x):(y)) -  #if defined(WORDS_BIGENDIAN)  # define BIT_ENDIAN_MACHINE 0  #else @@ -110,9 +101,6 @@ erts_init_bits(void)  {      ERTS_CT_ASSERT(offsetof(Binary,orig_bytes) % 8 == 0);      ERTS_CT_ASSERT(offsetof(ErtsMagicBinary,u.aligned.data) % 8 == 0); -    ERTS_CT_ASSERT(ERTS_MAGIC_BIN_BYTES_TO_ALIGN == -                   (offsetof(ErtsMagicBinary,u.aligned.data) -                    - offsetof(ErtsMagicBinary,u.unaligned.data)));      ERTS_CT_ASSERT(offsetof(ErtsBinary,driver.binary.orig_bytes)                  == offsetof(Binary,orig_bytes)); @@ -1324,7 +1312,14 @@ erts_bs_append(Process* c_p, Eterm* reg, Uint live, Eterm build_size_term,  	    goto badarg;  	}      } + +    if((ERTS_UINT_MAX - build_size_in_bits) < erts_bin_offset) { +        c_p->freason = SYSTEM_LIMIT; +        return THE_NON_VALUE; +    } +      used_size_in_bits = erts_bin_offset + build_size_in_bits; +      sb->is_writable = 0;	/* Make sure that no one else can write. */      pb->size = NBYTES(used_size_in_bits);      pb->flags |= PB_ACTIVE_WRITER; @@ -1398,16 +1393,27 @@ erts_bs_append(Process* c_p, Eterm* reg, Uint live, Eterm build_size_term,  		goto badarg;  	    }  	} -	used_size_in_bits = erts_bin_offset + build_size_in_bits; -	used_size_in_bytes = NBYTES(used_size_in_bits); -	bin_size = 2*used_size_in_bytes; + +        if((ERTS_UINT_MAX - build_size_in_bits) < erts_bin_offset) { +            c_p->freason = SYSTEM_LIMIT; +            return THE_NON_VALUE; +        } + +        used_size_in_bits = erts_bin_offset + build_size_in_bits; +        used_size_in_bytes = NBYTES(used_size_in_bits); + +        if(used_size_in_bits < (ERTS_UINT_MAX / 2)) { +            bin_size = 2 * used_size_in_bytes; +        } else { +            bin_size = NBYTES(ERTS_UINT_MAX); +        } +  	bin_size = (bin_size < 256) ? 256 : bin_size;  	/*  	 * Allocate the binary data struct itself.  	 */  	bptr = erts_bin_nrml_alloc(bin_size); -	erts_refc_init(&bptr->refc, 1);  	erts_current_bin = (byte *) bptr->orig_bytes;  	/* @@ -1491,6 +1497,12 @@ erts_bs_private_append(Process* p, Eterm bin, Eterm build_size_term, Uint unit)       * Calculate new size in bytes.       */      erts_bin_offset = 8*sb->size + sb->bitsize; + +    if((ERTS_UINT_MAX - build_size_in_bits) < erts_bin_offset) { +        p->freason = SYSTEM_LIMIT; +        return THE_NON_VALUE; +    } +      pos_in_bits_after_build = erts_bin_offset + build_size_in_bits;      pb->size = (pos_in_bits_after_build+7) >> 3;      pb->flags |= PB_ACTIVE_WRITER; @@ -1521,14 +1533,11 @@ erts_bs_private_append(Process* p, Eterm bin, Eterm build_size_term, Uint unit)  	     * binary and copy the contents of the old binary into it.  	     */  	    Binary* bptr = erts_bin_nrml_alloc(new_size); -	    erts_refc_init(&bptr->refc, 1);  	    sys_memcpy(bptr->orig_bytes, binp->orig_bytes, binp->orig_size);  	    pb->flags |= PB_IS_WRITABLE | PB_ACTIVE_WRITER;  	    pb->val = bptr;  	    pb->bytes = (byte *) bptr->orig_bytes; -	    if (erts_refc_dectest(&binp->refc, 0) == 0) { -		erts_bin_free(binp); -	    } +            erts_bin_release(binp);  	}      }      erts_current_bin = pb->bytes; @@ -1568,7 +1577,6 @@ erts_bs_init_writable(Process* p, Eterm sz)       * Allocate the binary data struct itself.       */      bptr = erts_bin_nrml_alloc(bin_size); -    erts_refc_init(&bptr->refc, 1);      /*       * Now allocate the ProcBin on the heap. @@ -1644,7 +1652,7 @@ erts_bs_get_unaligned_uint32(ErlBinMatchBuffer* mb)      return LSB[0] | (LSB[1]<<8) | (LSB[2]<<16) | (LSB[3]<<24);  } -void +static void  erts_align_utf8_bytes(ErlBinMatchBuffer* mb, byte* buf)  {      Uint bits = mb->size - mb->offset; | 
