📄 changes-table.c
字号:
/* changes-table.c : operations on the `changes' table
*
* ====================================================================
* Copyright (c) 2000-2004 CollabNet. All rights reserved.
*
* This software is licensed as described in the file COPYING, which
* you should have received as part of this distribution. The terms
* are also available at http://subversion.tigris.org/license-1.html.
* If newer versions of this license are posted there, you may use a
* newer version instead, at your option.
*
* This software consists of voluntary contributions made by many
* individuals. For exact contribution history, see the revision
* history and logs, available at http://subversion.tigris.org/.
* ====================================================================
*/
#include "bdb_compat.h"
#include <apr_hash.h>
#include <apr_tables.h>
#include "svn_fs.h"
#include "svn_pools.h"
#include "svn_path.h"
#include "../fs.h"
#include "../err.h"
#include "../trail.h"
#include "../id.h"
#include "../util/fs_skels.h"
#include "../../libsvn_fs/fs-loader.h"
#include "bdb-err.h"
#include "dbt.h"
#include "changes-table.h"
/*** Creating and opening the changes table. ***/
int
svn_fs_bdb__open_changes_table (DB **changes_p,
DB_ENV *env,
svn_boolean_t create)
{
const u_int32_t open_flags = (create ? (DB_CREATE | DB_EXCL) : 0);
DB *changes;
BDB_ERR (svn_fs_bdb__check_version());
BDB_ERR (db_create (&changes, env, 0));
/* Enable duplicate keys. This allows us to store the changes
one-per-row. Note: this must occur before ->open(). */
BDB_ERR (changes->set_flags (changes, DB_DUP));
BDB_ERR (changes->open (SVN_BDB_OPEN_PARAMS(changes, NULL),
"changes", 0, DB_BTREE,
open_flags | SVN_BDB_AUTO_COMMIT,
0666));
*changes_p = changes;
return 0;
}
/*** Storing and retrieving changes. ***/
svn_error_t *
svn_fs_bdb__changes_add (svn_fs_t *fs,
const char *key,
change_t *change,
trail_t *trail)
{
base_fs_data_t *bfd = fs->fsap_data;
DBT query, value;
skel_t *skel;
/* Convert native type to skel. */
SVN_ERR (svn_fs_base__unparse_change_skel (&skel, change, trail->pool));
/* Store a new record into the database. */
svn_fs_base__str_to_dbt (&query, key);
svn_fs_base__skel_to_dbt (&value, skel, trail->pool);
svn_fs_base__trail_debug (trail, "changes", "put");
SVN_ERR (BDB_WRAP (fs, "creating change",
bfd->changes->put (bfd->changes, trail->db_txn,
&query, &value, 0)));
return SVN_NO_ERROR;
}
svn_error_t *
svn_fs_bdb__changes_delete (svn_fs_t *fs,
const char *key,
trail_t *trail)
{
int db_err;
DBT query;
base_fs_data_t *bfd = fs->fsap_data;
svn_fs_base__trail_debug (trail, "changes", "del");
db_err = bfd->changes->del (bfd->changes, trail->db_txn,
svn_fs_base__str_to_dbt (&query, key), 0);
/* If there're no changes for KEY, that is acceptable. Any other
error should be propogated to the caller, though. */
if ((db_err) && (db_err != DB_NOTFOUND))
{
SVN_ERR (BDB_WRAP (fs, "deleting changes", db_err));
}
return SVN_NO_ERROR;
}
/* Merge the internal-use-only CHANGE into a hash of public-FS
svn_fs_path_change_t CHANGES, collapsing multiple changes into a
single summarical (is that real word?) change per path. */
static svn_error_t *
fold_change (apr_hash_t *changes,
const change_t *change)
{
apr_pool_t *pool = apr_hash_pool_get (changes);
svn_fs_path_change_t *old_change, *new_change;
const char *path;
if ((old_change = apr_hash_get (changes, change->path, APR_HASH_KEY_STRING)))
{
/* This path already exists in the hash, so we have to merge
this change into the already existing one. */
/* Since the path already exists in the hash, we don't have to
dup the allocation for the path itself. */
path = change->path;
/* Sanity check: only allow NULL node revision ID in the
`reset' case. */
if ((! change->noderev_id) && (change->kind != svn_fs_path_change_reset))
return svn_error_create
(SVN_ERR_FS_CORRUPT, NULL,
"Missing required node revision ID");
/* Sanity check: we should be talking about the same node
revision ID as our last change except where the last change
was a deletion. */
if (change->noderev_id
&& (! svn_fs_base__id_eq (old_change->node_rev_id,
change->noderev_id))
&& (old_change->change_kind != svn_fs_path_change_delete))
return svn_error_create
(SVN_ERR_FS_CORRUPT, NULL,
"Invalid change ordering: new node revision ID without delete");
/* Sanity check: an add, replacement, or reset must be the first
thing to follow a deletion. */
if ((old_change->change_kind == svn_fs_path_change_delete)
&& (! ((change->kind == svn_fs_path_change_replace)
|| (change->kind == svn_fs_path_change_reset)
|| (change->kind == svn_fs_path_change_add))))
return svn_error_create
(SVN_ERR_FS_CORRUPT, NULL,
"Invalid change ordering: non-add change on deleted path");
/* Now, merge that change in. */
switch (change->kind)
{
case svn_fs_path_change_reset:
/* A reset here will simply remove the path change from the
hash. */
old_change = NULL;
break;
case svn_fs_path_change_delete:
if (old_change->change_kind == svn_fs_path_change_add)
{
/* If the path was introduced in this transaction via an
add, and we are deleting it, just remove the path
altogether. */
old_change = NULL;
}
else
{
/* A deletion overrules all previous changes. */
old_change->change_kind = svn_fs_path_change_delete;
old_change->text_mod = change->text_mod;
old_change->prop_mod = change->prop_mod;
}
break;
case svn_fs_path_change_add:
case svn_fs_path_change_replace:
/* An add at this point must be following a previous delete,
so treat it just like a replace. */
old_change->change_kind = svn_fs_path_change_replace;
old_change->node_rev_id = svn_fs_base__id_copy (change->noderev_id,
pool);
old_change->text_mod = change->text_mod;
old_change->prop_mod = change->prop_mod;
break;
case svn_fs_path_change_modify:
default:
if (change->text_mod)
old_change->text_mod = TRUE;
if (change->prop_mod)
old_change->prop_mod = TRUE;
break;
}
/* Point our new_change to our (possibly modified) old_change. */
new_change = old_change;
}
else
{
/* This change is new to the hash, so make a new public change
structure from the internal one (in the hash's pool), and dup
the path into the hash's pool, too. */
new_change = apr_pcalloc (pool, sizeof (*new_change));
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -