diff options
Diffstat (limited to 'subversion/libsvn_fs_x/low_level.c')
| -rw-r--r-- | subversion/libsvn_fs_x/low_level.c | 195 | 
1 files changed, 158 insertions, 37 deletions
| diff --git a/subversion/libsvn_fs_x/low_level.c b/subversion/libsvn_fs_x/low_level.c index 76f3fd2f3872..5c7e3ad26610 100644 --- a/subversion/libsvn_fs_x/low_level.c +++ b/subversion/libsvn_fs_x/low_level.c @@ -1,4 +1,4 @@ -/* low_level.c --- low level r/w access to fs_x file structures +/* low_level.c --- low level r/w access to FSX file structures   *   * ====================================================================   *    Licensed to the Apache Software Foundation (ASF) under one @@ -56,7 +56,6 @@  #define ACTION_ADD         "add"  #define ACTION_DELETE      "delete"  #define ACTION_REPLACE     "replace" -#define ACTION_RESET       "reset"  /* True and False flags. */  #define FLAG_TRUE          "true" @@ -103,6 +102,19 @@ parse_revnum(svn_revnum_t *rev,    return SVN_NO_ERROR;  } +/* If ERR is not NULL, wrap it MESSAGE.  The latter must have an %ld + * format parameter that will be filled with REV. */ +static svn_error_t * +wrap_footer_error(svn_error_t *err, +                  const char *message, +                  svn_revnum_t rev) +{ +  if (err) +    return svn_error_quick_wrapf(err, message, rev); + +  return SVN_NO_ERROR; +} +  svn_error_t *  svn_fs_x__parse_footer(apr_off_t *l2p_offset,                         svn_checksum_t **l2p_checksum, @@ -110,6 +122,7 @@ svn_fs_x__parse_footer(apr_off_t *l2p_offset,                         svn_checksum_t **p2l_checksum,                         svn_stringbuf_t *footer,                         svn_revnum_t rev, +                       apr_off_t footer_offset,                         apr_pool_t *result_pool)  {    apr_int64_t val; @@ -118,17 +131,20 @@ svn_fs_x__parse_footer(apr_off_t *l2p_offset,    /* Get the L2P offset. */    const char *str = svn_cstring_tokenize(" ", &last_str);    if (str == NULL) -    return svn_error_create(SVN_ERR_FS_CORRUPT, NULL, -                            _("Invalid revision footer")); +    return svn_error_createf(SVN_ERR_FS_CORRUPT, NULL, +                             "Invalid r%ld footer", rev); -  SVN_ERR(svn_cstring_atoi64(&val, str)); +  SVN_ERR(wrap_footer_error(svn_cstring_strtoi64(&val, str, 0, +                                                 footer_offset - 1, 10), +                            "Invalid L2P offset in r%ld footer", +                            rev));    *l2p_offset = (apr_off_t)val;    /* Get the L2P checksum. */    str = svn_cstring_tokenize(" ", &last_str);    if (str == NULL) -    return svn_error_create(SVN_ERR_FS_CORRUPT, NULL, -                            _("Invalid revision footer")); +    return svn_error_createf(SVN_ERR_FS_CORRUPT, NULL, +                             "Invalid r%ld footer", rev);    SVN_ERR(svn_checksum_parse_hex(l2p_checksum, svn_checksum_md5, str,                                   result_pool)); @@ -136,17 +152,33 @@ svn_fs_x__parse_footer(apr_off_t *l2p_offset,    /* Get the P2L offset. */    str = svn_cstring_tokenize(" ", &last_str);    if (str == NULL) -    return svn_error_create(SVN_ERR_FS_CORRUPT, NULL, -                            _("Invalid revision footer")); +    return svn_error_createf(SVN_ERR_FS_CORRUPT, NULL, +                             "Invalid r%ld footer", rev); -  SVN_ERR(svn_cstring_atoi64(&val, str)); +  SVN_ERR(wrap_footer_error(svn_cstring_strtoi64(&val, str, 0, +                                                 footer_offset - 1, 10), +                            "Invalid P2L offset in r%ld footer", +                            rev));    *p2l_offset = (apr_off_t)val; +  /* The P2L indes follows the L2P index */ +  if (*p2l_offset <= *l2p_offset) +    return svn_error_createf(SVN_ERR_FS_CORRUPT, NULL, +                             "P2L offset %s must be larger than L2P offset %s" +                             " in r%ld footer", +                             apr_psprintf(result_pool, +                                          "%" APR_UINT64_T_HEX_FMT, +                                          (apr_uint64_t)*p2l_offset), +                             apr_psprintf(result_pool, +                                          "%" APR_UINT64_T_HEX_FMT, +                                          (apr_uint64_t)*l2p_offset), +                             rev); +    /* Get the P2L checksum. */    str = svn_cstring_tokenize(" ", &last_str);    if (str == NULL) -    return svn_error_create(SVN_ERR_FS_CORRUPT, NULL, -                            _("Invalid revision footer")); +    return svn_error_createf(SVN_ERR_FS_CORRUPT, NULL, +                             "Invalid r%ld footer", rev);    SVN_ERR(svn_checksum_parse_hex(p2l_checksum, svn_checksum_md5, str,                                   result_pool)); @@ -797,14 +829,6 @@ read_change(svn_fs_x__change_t **change_p,    change = apr_pcalloc(result_pool, sizeof(*change));    last_str = line->data; -  /* Get the node-id of the change. */ -  str = svn_cstring_tokenize(" ", &last_str); -  if (str == NULL) -    return svn_error_create(SVN_ERR_FS_CORRUPT, NULL, -                            _("Invalid changes line in rev-file")); - -  SVN_ERR(svn_fs_x__id_parse(&change->noderev_id, str)); -    /* Get the change type. */    str = svn_cstring_tokenize(" ", &last_str);    if (str == NULL) @@ -845,10 +869,6 @@ read_change(svn_fs_x__change_t **change_p,      {        change->change_kind = svn_fs_path_change_replace;      } -  else if (strcmp(str, ACTION_RESET) == 0) -    { -      change->change_kind = svn_fs_path_change_reset; -    }    else      {        return svn_error_create(SVN_ERR_FS_CORRUPT, NULL, @@ -954,10 +974,10 @@ read_change(svn_fs_x__change_t **change_p,  svn_error_t *  svn_fs_x__read_changes(apr_array_header_t **changes,                         svn_stream_t *stream, +                       int max_count,                         apr_pool_t *result_pool,                         apr_pool_t *scratch_pool)  { -  svn_fs_x__change_t *change;    apr_pool_t *iterpool;    /* Pre-allocate enough room for most change lists. @@ -970,13 +990,16 @@ svn_fs_x__read_changes(apr_array_header_t **changes,     */    *changes = apr_array_make(result_pool, 63, sizeof(svn_fs_x__change_t *)); -  SVN_ERR(read_change(&change, stream, result_pool, scratch_pool));    iterpool = svn_pool_create(scratch_pool); -  while (change) +  for (; max_count > 0; --max_count)      { -      APR_ARRAY_PUSH(*changes, svn_fs_x__change_t*) = change; -      SVN_ERR(read_change(&change, stream, result_pool, iterpool)); +      svn_fs_x__change_t *change;        svn_pool_clear(iterpool); +      SVN_ERR(read_change(&change, stream, result_pool, iterpool)); +      if (!change) +        break; +  +      APR_ARRAY_PUSH(*changes, svn_fs_x__change_t*) = change;      }    svn_pool_destroy(iterpool); @@ -1016,7 +1039,6 @@ write_change_entry(svn_stream_t *stream,                     svn_fs_x__change_t *change,                     apr_pool_t *scratch_pool)  { -  const char *idstr;    const char *change_string = NULL;    const char *kind_string = "";    svn_stringbuf_t *buf; @@ -1036,17 +1058,12 @@ write_change_entry(svn_stream_t *stream,      case svn_fs_path_change_replace:        change_string = ACTION_REPLACE;        break; -    case svn_fs_path_change_reset: -      change_string = ACTION_RESET; -      break;      default:        return svn_error_createf(SVN_ERR_FS_CORRUPT, NULL,                                 _("Invalid change type %d"),                                 change->change_kind);      } -  idstr = svn_fs_x__id_unparse(&change->noderev_id, scratch_pool)->data; -    SVN_ERR_ASSERT(change->node_kind == svn_node_dir                   || change->node_kind == svn_node_file);    kind_string = apr_psprintf(scratch_pool, "-%s", @@ -1054,8 +1071,8 @@ write_change_entry(svn_stream_t *stream,                               ? SVN_FS_X__KIND_DIR                               : SVN_FS_X__KIND_FILE); -  buf = svn_stringbuf_createf(scratch_pool, "%s %s%s %s %s %s %s\n", -                              idstr, change_string, kind_string, +  buf = svn_stringbuf_createf(scratch_pool, "%s%s %s %s %s %s\n", +                              change_string, kind_string,                                change->text_mod ? FLAG_TRUE : FLAG_FALSE,                                change->prop_mod ? FLAG_TRUE : FLAG_FALSE,                                change->mergeinfo_mod == svn_tristate_true @@ -1121,3 +1138,107 @@ svn_fs_x__write_changes(svn_stream_t *stream,    return SVN_NO_ERROR;  } +svn_error_t * +svn_fs_x__parse_properties(apr_hash_t **properties, +                           const svn_string_t *content, +                           apr_pool_t *result_pool) +{ +  const apr_byte_t *p = (const apr_byte_t *)content->data; +  const apr_byte_t *end = p + content->len; +  apr_uint64_t count; + +  *properties = apr_hash_make(result_pool); + +  /* Extract the number of properties we are expected to read. */ +  p = svn__decode_uint(&count, p, end); + +  /* Read all the properties we find. +     Because prop-name and prop-value are nicely NUL-terminated +     sub-strings of CONTENT, we can simply reference them there. +     I.e. there is no need to copy them around. +   */ +  while (p < end) +    { +      apr_uint64_t value_len; +      svn_string_t *value; + +      const char *key = (const char *)p; + +      /* Note that this may never overflow / segfault because +         CONTENT itself is NUL-terminated. */ +      apr_size_t key_len = strlen(key); +      p += key_len + 1; +      if (key[key_len]) +        return svn_error_createf(SVN_ERR_FS_CORRUPT_PROPLIST, NULL, +                                 "Property name not NUL terminated"); + +      if (p >= end) +        return svn_error_createf(SVN_ERR_FS_CORRUPT_PROPLIST, NULL, +                                 "Property value missing"); +      p = svn__decode_uint(&value_len, p, end); +      if (value_len >= (end - p)) +        return svn_error_createf(SVN_ERR_FS_CORRUPT_PROPLIST, NULL, +                                 "Property value too long"); + +      value = apr_pcalloc(result_pool, sizeof(*value)); +      value->data = (const char *)p; +      value->len = (apr_size_t)value_len; +      if (p[value->len]) +        return svn_error_createf(SVN_ERR_FS_CORRUPT_PROPLIST, NULL, +                                 "Property value not NUL terminated"); + +      p += value->len + 1; + +      apr_hash_set(*properties, key, key_len, value); +    } + +  /* Check that we read the expected number of properties. */ +  if ((apr_uint64_t)apr_hash_count(*properties) != count) +    return svn_error_createf(SVN_ERR_FS_CORRUPT_PROPLIST, NULL, +                             "Property count mismatch"); + +  return SVN_NO_ERROR; +} + +svn_error_t * +svn_fs_x__write_properties(svn_stream_t *stream, +                           apr_hash_t *proplist, +                           apr_pool_t *scratch_pool) +{ +  apr_byte_t buffer[SVN__MAX_ENCODED_UINT_LEN]; +  apr_size_t len; +  apr_hash_index_t *hi; + +  /* Write the number of properties in this list. */ +  len = svn__encode_uint(buffer, apr_hash_count(proplist)) - buffer; +  SVN_ERR(svn_stream_write(stream, (const char *)buffer, &len)); + +  /* Serialize each property as follows: +     <Prop-name> <NUL> +     <Value-len> <Prop-value> <NUL> +   */ +  for (hi = apr_hash_first(scratch_pool, proplist); +       hi; +       hi = apr_hash_next(hi)) +    { +      const char *key; +      apr_size_t key_len; +      svn_string_t *value; +      apr_hash_this(hi, (const void **)&key, (apr_ssize_t *)&key_len, +                    (void **)&value); + +      /* Include the terminating NUL. */ +      ++key_len; +      SVN_ERR(svn_stream_write(stream, key, &key_len)); + +      len = svn__encode_uint(buffer, value->len) - buffer; +      SVN_ERR(svn_stream_write(stream, (const char *)buffer, &len)); +      SVN_ERR(svn_stream_write(stream, value->data, &value->len)); + +      /* Terminate with NUL. */ +      len = 1; +      SVN_ERR(svn_stream_write(stream, "", &len)); +    } + +  return SVN_NO_ERROR; +} | 
