Blame src/diff_file.c

Packit ae9e2a
/*
Packit ae9e2a
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit ae9e2a
 *
Packit ae9e2a
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit ae9e2a
 * a Linking Exception. For full terms see the included COPYING file.
Packit ae9e2a
 */
Packit ae9e2a
#include "common.h"
Packit ae9e2a
#include "git2/blob.h"
Packit ae9e2a
#include "git2/submodule.h"
Packit ae9e2a
#include "diff.h"
Packit ae9e2a
#include "diff_generate.h"
Packit ae9e2a
#include "diff_file.h"
Packit ae9e2a
#include "odb.h"
Packit ae9e2a
#include "fileops.h"
Packit ae9e2a
#include "filter.h"
Packit ae9e2a
Packit ae9e2a
#define DIFF_MAX_FILESIZE 0x20000000
Packit ae9e2a
Packit ae9e2a
static bool diff_file_content_binary_by_size(git_diff_file_content *fc)
Packit ae9e2a
{
Packit ae9e2a
	/* if we have diff opts, check max_size vs file size */
Packit ae9e2a
	if ((fc->file->flags & DIFF_FLAGS_KNOWN_BINARY) == 0 &&
Packit ae9e2a
		fc->opts_max_size > 0 &&
Packit ae9e2a
		fc->file->size > fc->opts_max_size)
Packit ae9e2a
		fc->file->flags |= GIT_DIFF_FLAG_BINARY;
Packit ae9e2a
Packit ae9e2a
	return ((fc->file->flags & GIT_DIFF_FLAG_BINARY) != 0);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static void diff_file_content_binary_by_content(git_diff_file_content *fc)
Packit ae9e2a
{
Packit ae9e2a
	if ((fc->file->flags & DIFF_FLAGS_KNOWN_BINARY) != 0)
Packit ae9e2a
		return;
Packit ae9e2a
Packit ae9e2a
	switch (git_diff_driver_content_is_binary(
Packit ae9e2a
		fc->driver, fc->map.data, fc->map.len)) {
Packit ae9e2a
	case 0: fc->file->flags |= GIT_DIFF_FLAG_NOT_BINARY; break;
Packit ae9e2a
	case 1: fc->file->flags |= GIT_DIFF_FLAG_BINARY; break;
Packit ae9e2a
	default: break;
Packit ae9e2a
	}
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int diff_file_content_init_common(
Packit ae9e2a
	git_diff_file_content *fc, const git_diff_options *opts)
Packit ae9e2a
{
Packit ae9e2a
	fc->opts_flags = opts ? opts->flags : GIT_DIFF_NORMAL;
Packit ae9e2a
Packit ae9e2a
	if (opts && opts->max_size >= 0)
Packit ae9e2a
		fc->opts_max_size = opts->max_size ?
Packit ae9e2a
			opts->max_size : DIFF_MAX_FILESIZE;
Packit ae9e2a
Packit ae9e2a
	if (fc->src == GIT_ITERATOR_TYPE_EMPTY)
Packit ae9e2a
		fc->src = GIT_ITERATOR_TYPE_TREE;
Packit ae9e2a
Packit ae9e2a
	if (!fc->driver &&
Packit ae9e2a
		git_diff_driver_lookup(&fc->driver, fc->repo, fc->file->path) < 0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	/* give driver a chance to modify options */
Packit ae9e2a
	git_diff_driver_update_options(&fc->opts_flags, fc->driver);
Packit ae9e2a
Packit ae9e2a
	/* make sure file is conceivable mmap-able */
Packit ae9e2a
	if ((git_off_t)((size_t)fc->file->size) != fc->file->size)
Packit ae9e2a
		fc->file->flags |= GIT_DIFF_FLAG_BINARY;
Packit ae9e2a
	/* check if user is forcing text diff the file */
Packit ae9e2a
	else if (fc->opts_flags & GIT_DIFF_FORCE_TEXT) {
Packit ae9e2a
		fc->file->flags &= ~GIT_DIFF_FLAG_BINARY;
Packit ae9e2a
		fc->file->flags |= GIT_DIFF_FLAG_NOT_BINARY;
Packit ae9e2a
	}
Packit ae9e2a
	/* check if user is forcing binary diff the file */
Packit ae9e2a
	else if (fc->opts_flags & GIT_DIFF_FORCE_BINARY) {
Packit ae9e2a
		fc->file->flags &= ~GIT_DIFF_FLAG_NOT_BINARY;
Packit ae9e2a
		fc->file->flags |= GIT_DIFF_FLAG_BINARY;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	diff_file_content_binary_by_size(fc);
Packit ae9e2a
Packit ae9e2a
	if ((fc->flags & GIT_DIFF_FLAG__NO_DATA) != 0) {
Packit ae9e2a
		fc->flags |= GIT_DIFF_FLAG__LOADED;
Packit ae9e2a
		fc->map.len  = 0;
Packit ae9e2a
		fc->map.data = "";
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if ((fc->flags & GIT_DIFF_FLAG__LOADED) != 0)
Packit ae9e2a
		diff_file_content_binary_by_content(fc);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_diff_file_content__init_from_diff(
Packit ae9e2a
	git_diff_file_content *fc,
Packit ae9e2a
	git_diff *diff,
Packit ae9e2a
	git_diff_delta *delta,
Packit ae9e2a
	bool use_old)
Packit ae9e2a
{
Packit ae9e2a
	bool has_data = true;
Packit ae9e2a
Packit ae9e2a
	memset(fc, 0, sizeof(*fc));
Packit ae9e2a
	fc->repo = diff->repo;
Packit ae9e2a
	fc->file = use_old ? &delta->old_file : &delta->new_file;
Packit ae9e2a
	fc->src  = use_old ? diff->old_src : diff->new_src;
Packit ae9e2a
Packit ae9e2a
	if (git_diff_driver_lookup(&fc->driver, fc->repo, fc->file->path) < 0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	switch (delta->status) {
Packit ae9e2a
	case GIT_DELTA_ADDED:
Packit ae9e2a
		has_data = !use_old; break;
Packit ae9e2a
	case GIT_DELTA_DELETED:
Packit ae9e2a
		has_data = use_old; break;
Packit ae9e2a
	case GIT_DELTA_UNTRACKED:
Packit ae9e2a
		has_data = !use_old &&
Packit ae9e2a
			(diff->opts.flags & GIT_DIFF_SHOW_UNTRACKED_CONTENT) != 0;
Packit ae9e2a
		break;
Packit ae9e2a
	case GIT_DELTA_UNREADABLE:
Packit ae9e2a
	case GIT_DELTA_MODIFIED:
Packit ae9e2a
	case GIT_DELTA_COPIED:
Packit ae9e2a
	case GIT_DELTA_RENAMED:
Packit ae9e2a
		break;
Packit ae9e2a
	default:
Packit ae9e2a
		has_data = false;
Packit ae9e2a
		break;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (!has_data)
Packit ae9e2a
		fc->flags |= GIT_DIFF_FLAG__NO_DATA;
Packit ae9e2a
Packit ae9e2a
	return diff_file_content_init_common(fc, &diff->opts);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_diff_file_content__init_from_src(
Packit ae9e2a
	git_diff_file_content *fc,
Packit ae9e2a
	git_repository *repo,
Packit ae9e2a
	const git_diff_options *opts,
Packit ae9e2a
	const git_diff_file_content_src *src,
Packit ae9e2a
	git_diff_file *as_file)
Packit ae9e2a
{
Packit ae9e2a
	memset(fc, 0, sizeof(*fc));
Packit ae9e2a
	fc->repo = repo;
Packit ae9e2a
	fc->file = as_file;
Packit ae9e2a
Packit ae9e2a
	if (!src->blob && !src->buf) {
Packit ae9e2a
		fc->flags |= GIT_DIFF_FLAG__NO_DATA;
Packit ae9e2a
	} else {
Packit ae9e2a
		fc->flags |= GIT_DIFF_FLAG__LOADED;
Packit ae9e2a
		fc->file->flags |= GIT_DIFF_FLAG_VALID_ID;
Packit ae9e2a
		fc->file->mode = GIT_FILEMODE_BLOB;
Packit ae9e2a
Packit ae9e2a
		if (src->blob) {
Packit ae9e2a
			git_blob_dup((git_blob **)&fc->blob, (git_blob *) src->blob);
Packit ae9e2a
			fc->file->size = git_blob_rawsize(src->blob);
Packit ae9e2a
			git_oid_cpy(&fc->file->id, git_blob_id(src->blob));
Packit ae9e2a
			fc->file->id_abbrev = GIT_OID_HEXSZ;
Packit ae9e2a
Packit ae9e2a
			fc->map.len  = (size_t)fc->file->size;
Packit ae9e2a
			fc->map.data = (char *)git_blob_rawcontent(src->blob);
Packit ae9e2a
Packit ae9e2a
			fc->flags |= GIT_DIFF_FLAG__FREE_BLOB;
Packit ae9e2a
		} else {
Packit ae9e2a
			fc->file->size = src->buflen;
Packit ae9e2a
			git_odb_hash(&fc->file->id, src->buf, src->buflen, GIT_OBJ_BLOB);
Packit ae9e2a
			fc->file->id_abbrev = GIT_OID_HEXSZ;
Packit ae9e2a
Packit ae9e2a
			fc->map.len  = src->buflen;
Packit ae9e2a
			fc->map.data = (char *)src->buf;
Packit ae9e2a
		}
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	return diff_file_content_init_common(fc, opts);
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int diff_file_content_commit_to_str(
Packit ae9e2a
	git_diff_file_content *fc, bool check_status)
Packit ae9e2a
{
Packit ae9e2a
	char oid[GIT_OID_HEXSZ+1];
Packit ae9e2a
	git_buf content = GIT_BUF_INIT;
Packit ae9e2a
	const char *status = "";
Packit ae9e2a
Packit ae9e2a
	if (check_status) {
Packit ae9e2a
		int error = 0;
Packit ae9e2a
		git_submodule *sm = NULL;
Packit ae9e2a
		unsigned int sm_status = 0;
Packit ae9e2a
		const git_oid *sm_head;
Packit ae9e2a
Packit ae9e2a
		if ((error = git_submodule_lookup(&sm, fc->repo, fc->file->path)) < 0) {
Packit ae9e2a
			/* GIT_EEXISTS means a "submodule" that has not been git added */
Packit ae9e2a
			if (error == GIT_EEXISTS) {
Packit ae9e2a
				giterr_clear();
Packit ae9e2a
				error = 0;
Packit ae9e2a
			}
Packit ae9e2a
			return error;
Packit ae9e2a
		}
Packit ae9e2a
Packit ae9e2a
		if ((error = git_submodule_status(&sm_status, fc->repo, fc->file->path, GIT_SUBMODULE_IGNORE_UNSPECIFIED)) < 0) {
Packit ae9e2a
			git_submodule_free(sm);
Packit ae9e2a
			return error;
Packit ae9e2a
		}
Packit ae9e2a
Packit ae9e2a
		/* update OID if we didn't have it previously */
Packit ae9e2a
		if ((fc->file->flags & GIT_DIFF_FLAG_VALID_ID) == 0 &&
Packit ae9e2a
			((sm_head = git_submodule_wd_id(sm)) != NULL ||
Packit ae9e2a
			 (sm_head = git_submodule_head_id(sm)) != NULL))
Packit ae9e2a
		{
Packit ae9e2a
			git_oid_cpy(&fc->file->id, sm_head);
Packit ae9e2a
			fc->file->flags |= GIT_DIFF_FLAG_VALID_ID;
Packit ae9e2a
		}
Packit ae9e2a
Packit ae9e2a
		if (GIT_SUBMODULE_STATUS_IS_WD_DIRTY(sm_status))
Packit ae9e2a
			status = "-dirty";
Packit ae9e2a
Packit ae9e2a
		git_submodule_free(sm);
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	git_oid_tostr(oid, sizeof(oid), &fc->file->id);
Packit ae9e2a
	if (git_buf_printf(&content, "Subproject commit %s%s\n", oid, status) < 0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	fc->map.len  = git_buf_len(&content);
Packit ae9e2a
	fc->map.data = git_buf_detach(&content);
Packit ae9e2a
	fc->flags |= GIT_DIFF_FLAG__FREE_DATA;
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int diff_file_content_load_blob(
Packit ae9e2a
	git_diff_file_content *fc,
Packit ae9e2a
	git_diff_options *opts)
Packit ae9e2a
{
Packit ae9e2a
	int error = 0;
Packit ae9e2a
	git_odb_object *odb_obj = NULL;
Packit ae9e2a
Packit ae9e2a
	if (git_oid_iszero(&fc->file->id))
Packit ae9e2a
		return 0;
Packit ae9e2a
Packit ae9e2a
	if (fc->file->mode == GIT_FILEMODE_COMMIT)
Packit ae9e2a
		return diff_file_content_commit_to_str(fc, false);
Packit ae9e2a
Packit ae9e2a
	/* if we don't know size, try to peek at object header first */
Packit ae9e2a
	if (!fc->file->size) {
Packit ae9e2a
		if ((error = git_diff_file__resolve_zero_size(
Packit ae9e2a
				fc->file, &odb_obj, fc->repo)) < 0)
Packit ae9e2a
			return error;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if ((opts->flags & GIT_DIFF_SHOW_BINARY) == 0 &&
Packit ae9e2a
		diff_file_content_binary_by_size(fc))
Packit ae9e2a
		return 0;
Packit ae9e2a
Packit ae9e2a
	if (odb_obj != NULL) {
Packit ae9e2a
		error = git_object__from_odb_object(
Packit ae9e2a
			(git_object **)&fc->blob, fc->repo, odb_obj, GIT_OBJ_BLOB);
Packit ae9e2a
		git_odb_object_free(odb_obj);
Packit ae9e2a
	} else {
Packit ae9e2a
		error = git_blob_lookup(
Packit ae9e2a
			(git_blob **)&fc->blob, fc->repo, &fc->file->id);
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (!error) {
Packit ae9e2a
		fc->flags |= GIT_DIFF_FLAG__FREE_BLOB;
Packit ae9e2a
		fc->map.data = (void *)git_blob_rawcontent(fc->blob);
Packit ae9e2a
		fc->map.len  = (size_t)git_blob_rawsize(fc->blob);
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int diff_file_content_load_workdir_symlink_fake(
Packit ae9e2a
	git_diff_file_content *fc, git_buf *path)
Packit ae9e2a
{
Packit ae9e2a
	git_buf target = GIT_BUF_INIT;
Packit ae9e2a
	int error;
Packit ae9e2a
Packit ae9e2a
	if ((error = git_futils_readbuffer(&target, path->ptr)) < 0)
Packit ae9e2a
		return error;
Packit ae9e2a
Packit ae9e2a
	fc->map.len = git_buf_len(&target);
Packit ae9e2a
	fc->map.data = git_buf_detach(&target);
Packit ae9e2a
	fc->flags |= GIT_DIFF_FLAG__FREE_DATA;
Packit ae9e2a
Packit ae9e2a
	git_buf_free(&target);
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int diff_file_content_load_workdir_symlink(
Packit ae9e2a
	git_diff_file_content *fc, git_buf *path)
Packit ae9e2a
{
Packit ae9e2a
	ssize_t alloc_len, read_len;
Packit ae9e2a
	int symlink_supported, error;
Packit ae9e2a
Packit ae9e2a
	if ((error = git_repository__cvar(
Packit ae9e2a
		&symlink_supported, fc->repo, GIT_CVAR_SYMLINKS)) < 0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	if (!symlink_supported)
Packit ae9e2a
		return diff_file_content_load_workdir_symlink_fake(fc, path);
Packit ae9e2a
Packit ae9e2a
	/* link path on disk could be UTF-16, so prepare a buffer that is
Packit ae9e2a
	 * big enough to handle some UTF-8 data expansion
Packit ae9e2a
	 */
Packit ae9e2a
	alloc_len = (ssize_t)(fc->file->size * 2) + 1;
Packit ae9e2a
Packit ae9e2a
	fc->map.data = git__calloc(alloc_len, sizeof(char));
Packit ae9e2a
	GITERR_CHECK_ALLOC(fc->map.data);
Packit ae9e2a
Packit ae9e2a
	fc->flags |= GIT_DIFF_FLAG__FREE_DATA;
Packit ae9e2a
Packit ae9e2a
	read_len = p_readlink(git_buf_cstr(path), fc->map.data, alloc_len);
Packit ae9e2a
	if (read_len < 0) {
Packit ae9e2a
		giterr_set(GITERR_OS, "failed to read symlink '%s'", fc->file->path);
Packit ae9e2a
		return -1;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	fc->map.len = read_len;
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int diff_file_content_load_workdir_file(
Packit ae9e2a
	git_diff_file_content *fc,
Packit ae9e2a
	git_buf *path,
Packit ae9e2a
	git_diff_options *diff_opts)
Packit ae9e2a
{
Packit ae9e2a
	int error = 0;
Packit ae9e2a
	git_filter_list *fl = NULL;
Packit ae9e2a
	git_file fd = git_futils_open_ro(git_buf_cstr(path));
Packit ae9e2a
	git_buf raw = GIT_BUF_INIT;
Packit ae9e2a
Packit ae9e2a
	if (fd < 0)
Packit ae9e2a
		return fd;
Packit ae9e2a
Packit ae9e2a
	if (!fc->file->size &&
Packit ae9e2a
		!(fc->file->size = git_futils_filesize(fd)))
Packit ae9e2a
		goto cleanup;
Packit ae9e2a
Packit ae9e2a
	if ((diff_opts->flags & GIT_DIFF_SHOW_BINARY) == 0 &&
Packit ae9e2a
		diff_file_content_binary_by_size(fc))
Packit ae9e2a
		goto cleanup;
Packit ae9e2a
Packit ae9e2a
	if ((error = git_filter_list_load(
Packit ae9e2a
			&fl, fc->repo, NULL, fc->file->path,
Packit ae9e2a
			GIT_FILTER_TO_ODB, GIT_FILTER_ALLOW_UNSAFE)) < 0)
Packit ae9e2a
		goto cleanup;
Packit ae9e2a
Packit ae9e2a
	/* if there are no filters, try to mmap the file */
Packit ae9e2a
	if (fl == NULL) {
Packit ae9e2a
		if (!(error = git_futils_mmap_ro(
Packit ae9e2a
				&fc->map, fd, 0, (size_t)fc->file->size))) {
Packit ae9e2a
			fc->flags |= GIT_DIFF_FLAG__UNMAP_DATA;
Packit ae9e2a
			goto cleanup;
Packit ae9e2a
		}
Packit ae9e2a
Packit ae9e2a
		/* if mmap failed, fall through to try readbuffer below */
Packit ae9e2a
		giterr_clear();
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (!(error = git_futils_readbuffer_fd(&raw, fd, (size_t)fc->file->size))) {
Packit ae9e2a
		git_buf out = GIT_BUF_INIT;
Packit ae9e2a
Packit ae9e2a
		error = git_filter_list_apply_to_data(&out, fl, &raw;;
Packit ae9e2a
Packit ae9e2a
		if (out.ptr != raw.ptr)
Packit ae9e2a
			git_buf_free(&raw;;
Packit ae9e2a
Packit ae9e2a
		if (!error) {
Packit ae9e2a
			fc->map.len  = out.size;
Packit ae9e2a
			fc->map.data = out.ptr;
Packit ae9e2a
			fc->flags |= GIT_DIFF_FLAG__FREE_DATA;
Packit ae9e2a
		}
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
cleanup:
Packit ae9e2a
	git_filter_list_free(fl);
Packit ae9e2a
	p_close(fd);
Packit ae9e2a
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
static int diff_file_content_load_workdir(
Packit ae9e2a
	git_diff_file_content *fc,
Packit ae9e2a
	git_diff_options *diff_opts)
Packit ae9e2a
{
Packit ae9e2a
	int error = 0;
Packit ae9e2a
	git_buf path = GIT_BUF_INIT;
Packit ae9e2a
Packit ae9e2a
	if (fc->file->mode == GIT_FILEMODE_COMMIT)
Packit ae9e2a
		return diff_file_content_commit_to_str(fc, true);
Packit ae9e2a
Packit ae9e2a
	if (fc->file->mode == GIT_FILEMODE_TREE)
Packit ae9e2a
		return 0;
Packit ae9e2a
Packit ae9e2a
	if (git_buf_joinpath(
Packit ae9e2a
			&path, git_repository_workdir(fc->repo), fc->file->path) < 0)
Packit ae9e2a
		return -1;
Packit ae9e2a
Packit ae9e2a
	if (S_ISLNK(fc->file->mode))
Packit ae9e2a
		error = diff_file_content_load_workdir_symlink(fc, &path);
Packit ae9e2a
	else
Packit ae9e2a
		error = diff_file_content_load_workdir_file(fc, &path, diff_opts);
Packit ae9e2a
Packit ae9e2a
	/* once data is loaded, update OID if we didn't have it previously */
Packit ae9e2a
	if (!error && (fc->file->flags & GIT_DIFF_FLAG_VALID_ID) == 0) {
Packit ae9e2a
		error = git_odb_hash(
Packit ae9e2a
			&fc->file->id, fc->map.data, fc->map.len, GIT_OBJ_BLOB);
Packit ae9e2a
		fc->file->flags |= GIT_DIFF_FLAG_VALID_ID;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	git_buf_free(&path);
Packit ae9e2a
	return error;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
int git_diff_file_content__load(
Packit ae9e2a
	git_diff_file_content *fc,
Packit ae9e2a
	git_diff_options *diff_opts)
Packit ae9e2a
{
Packit ae9e2a
	int error = 0;
Packit ae9e2a
Packit ae9e2a
	if ((fc->flags & GIT_DIFF_FLAG__LOADED) != 0)
Packit ae9e2a
		return 0;
Packit ae9e2a
Packit ae9e2a
	if ((fc->file->flags & GIT_DIFF_FLAG_BINARY) != 0 &&
Packit ae9e2a
		(diff_opts->flags & GIT_DIFF_SHOW_BINARY) == 0)
Packit ae9e2a
		return 0;
Packit ae9e2a
Packit ae9e2a
	if (fc->src == GIT_ITERATOR_TYPE_WORKDIR)
Packit ae9e2a
		error = diff_file_content_load_workdir(fc, diff_opts);
Packit ae9e2a
	else
Packit ae9e2a
		error = diff_file_content_load_blob(fc, diff_opts);
Packit ae9e2a
	if (error)
Packit ae9e2a
		return error;
Packit ae9e2a
Packit ae9e2a
	fc->flags |= GIT_DIFF_FLAG__LOADED;
Packit ae9e2a
Packit ae9e2a
	diff_file_content_binary_by_content(fc);
Packit ae9e2a
Packit ae9e2a
	return 0;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void git_diff_file_content__unload(git_diff_file_content *fc)
Packit ae9e2a
{
Packit ae9e2a
	if ((fc->flags & GIT_DIFF_FLAG__LOADED) == 0)
Packit ae9e2a
		return;
Packit ae9e2a
Packit ae9e2a
	if (fc->flags & GIT_DIFF_FLAG__FREE_DATA) {
Packit ae9e2a
		git__free(fc->map.data);
Packit ae9e2a
		fc->map.data = "";
Packit ae9e2a
		fc->map.len  = 0;
Packit ae9e2a
		fc->flags &= ~GIT_DIFF_FLAG__FREE_DATA;
Packit ae9e2a
	}
Packit ae9e2a
	else if (fc->flags & GIT_DIFF_FLAG__UNMAP_DATA) {
Packit ae9e2a
		git_futils_mmap_free(&fc->map);
Packit ae9e2a
		fc->map.data = "";
Packit ae9e2a
		fc->map.len  = 0;
Packit ae9e2a
		fc->flags &= ~GIT_DIFF_FLAG__UNMAP_DATA;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	if (fc->flags & GIT_DIFF_FLAG__FREE_BLOB) {
Packit ae9e2a
		git_blob_free((git_blob *)fc->blob);
Packit ae9e2a
		fc->blob = NULL;
Packit ae9e2a
		fc->flags &= ~GIT_DIFF_FLAG__FREE_BLOB;
Packit ae9e2a
	}
Packit ae9e2a
Packit ae9e2a
	fc->flags &= ~GIT_DIFF_FLAG__LOADED;
Packit ae9e2a
}
Packit ae9e2a
Packit ae9e2a
void git_diff_file_content__clear(git_diff_file_content *fc)
Packit ae9e2a
{
Packit ae9e2a
	git_diff_file_content__unload(fc);
Packit ae9e2a
Packit ae9e2a
	/* for now, nothing else to do */
Packit ae9e2a
}