[svn.haxx.se] · SVN Dev · SVN Users · SVN Org · TSVN Dev · TSVN Users · Subclipse Dev · Subclipse Users · this month's index

[PATCH] Fix crash in svn_wc_get_status_editor2

From: Julian Foad <julianfoad_at_btopenworld.com>
Date: 2005-11-29 14:52:02 CET

I've encountered a bug while reviewing Fabien Coelho's "svnversion to
libsvn_wc" patch.

  * If @a traversal_info is non-null, then record pre-update traversal
  * state in it. (Caller should obtain @a traversal_info from
  * svn_wc_init_traversal_info().)
...
svn_error_t *
svn_wc_get_status_editor2 (const svn_delta_editor_t **editor,
                            ...
                            svn_wc_traversal_info_t *traversal_info, ...)

It crashes if I pass NULL for traversal_info. The only caller we have so far
doesn't pass null, but the new one will want to.

This patch fixes it. Is it right?

There is a possible alternative: Change the doc string to require that this
parameter is provided. We might be able to justify that API change as a bug
fix, because nobody can be using it much because it doesn't work, at least on
WCs with unversioned items in them. But that's not really a safe attitude, and
anyway it's ugly to require it from a caller that doesn't want it. (Revving
the API isn't really a solution: we need to fix the bug in this version.)

- Julian

Make svn_wc_get_status_editor2() honour its API promise to work even if its
'traversal_info' argument is null. We haven't yet called it in such a way.

* subversion/libsvn_wc/status.c
  (get_dir_status):
    Don't let a null traversal_info get in the way of handling externals.
  (close_edit): Don't try to modify traversal_info if it's null.
  (svn_wc_get_status_editor2):
    Make the local 'externals' hash independent of traversal_info.

Index: subversion/libsvn_wc/status.c
===================================================================
--- subversion/libsvn_wc/status.c (revision 17521)
+++ subversion/libsvn_wc/status.c (working copy)
@@ -809,31 +809,34 @@ get_dir_status (struct edit_baton *eb,
   /* If "this dir" has "svn:externals" property set on it, store its
      name and value in traversal_info. Also, we want to track the
      externals internally so we can report status more accurately. */
- if (eb->traversal_info)
     {
       const svn_string_t *prop_val;
       SVN_ERR (svn_wc_prop_get (&prop_val, SVN_PROP_EXTERNALS, path,
                                 adm_access, subpool));
       if (prop_val)
         {
- apr_pool_t *dup_pool = eb->traversal_info->pool;
- const char *dup_path = apr_pstrdup (dup_pool, path);
- const char *dup_val = apr_pstrmemdup (dup_pool, prop_val->data,
- prop_val->len);
           apr_array_header_t *ext_items;
           int i;
 
- /* First things first -- we put the externals information
- into the "global" traversal info structure. */
- apr_hash_set (eb->traversal_info->externals_old,
- dup_path, APR_HASH_KEY_STRING, dup_val);
- apr_hash_set (eb->traversal_info->externals_new,
- dup_path, APR_HASH_KEY_STRING, dup_val);
+ if (eb->traversal_info)
+ {
+ apr_pool_t *dup_pool = eb->traversal_info->pool;
+ const char *dup_path = apr_pstrdup (dup_pool, path);
+ const char *dup_val = apr_pstrmemdup (dup_pool, prop_val->data,
+ prop_val->len);
+
+ /* First things first -- we put the externals information
+ into the "global" traversal info structure. */
+ apr_hash_set (eb->traversal_info->externals_old,
+ dup_path, APR_HASH_KEY_STRING, dup_val);
+ apr_hash_set (eb->traversal_info->externals_new,
+ dup_path, APR_HASH_KEY_STRING, dup_val);
+ }
 
           /* Now, parse the thing, and copy the parsed results into
              our "global" externals hash. */
           SVN_ERR (svn_wc_parse_externals_description2 (&ext_items, path,
- dup_val, dup_pool));
+ prop_val->data, pool));
           for (i = 0; ext_items && i < ext_items->nelts; i++)
             {
               svn_wc_external_item_t *item;
@@ -841,7 +844,7 @@ get_dir_status (struct edit_baton *eb,
               item = APR_ARRAY_IDX (ext_items, i, svn_wc_external_item_t *);
               apr_hash_set (eb->externals, svn_path_join (path,
                                                           item->target_dir,
- dup_pool),
+ pool),
                             APR_HASH_KEY_STRING, item);
             }
         }
@@ -1852,7 +1855,7 @@ close_edit (void *edit_baton,
  cleanup:
   /* Let's make sure that we didn't harvest any traversal info for the
      anchor if we had a target. */
- if (*eb->target)
+ if (eb->traversal_info && *eb->target)
     {
       apr_hash_set (eb->traversal_info->externals_old,
                     eb->anchor, APR_HASH_KEY_STRING, NULL);
@@ -1901,9 +1904,7 @@ svn_wc_get_status_editor2 (const svn_del
   eb->cancel_func = cancel_func;
   eb->cancel_baton = cancel_baton;
   eb->traversal_info = traversal_info;
- eb->externals = traversal_info
- ? apr_hash_make (traversal_info->pool)
- : NULL;
+ eb->externals = apr_hash_make (pool);
   eb->anchor = svn_wc_adm_access_path (anchor);
   eb->target = target;
   eb->root_opened = FALSE;

---------------------------------------------------------------------
To unsubscribe, e-mail: dev-unsubscribe@subversion.tigris.org
For additional commands, e-mail: dev-help@subversion.tigris.org
Received on Tue Nov 29 14:56:31 2005

This is an archived mail posted to the Subversion Dev mailing list.

This site is subject to the Apache Privacy Policy and the Apache Public Forum Archive Policy.