Blame gfs2/libgfs2/buf.c

Packit Service 360c39
#include "clusterautoconfig.h"
Packit Service 360c39
Packit Service 360c39
#include <stdio.h>
Packit Service 360c39
#include <stdlib.h>
Packit Service 360c39
#include <string.h>
Packit Service 360c39
#include <stdint.h>
Packit Service 360c39
#include <inttypes.h>
Packit Service 360c39
#include <sys/types.h>
Packit Service 360c39
#include <sys/stat.h>
Packit Service 360c39
#include <sys/time.h>
Packit Service 360c39
#include <fcntl.h>
Packit Service 360c39
#include <unistd.h>
Packit Service 360c39
#include <errno.h>
Packit Service 360c39
#include <linux/types.h>
Packit Service 360c39
Packit Service 360c39
#include "libgfs2.h"
Packit Service 360c39
Packit Service 360c39
#ifndef IOV_MAX
Packit Service 360c39
  #ifdef UIO_MAXIOV
Packit Service 360c39
    #define IOV_MAX UIO_MAXIOV
Packit Service 360c39
  #else
Packit Service 360c39
    #define IOV_MAX (1024)
Packit Service 360c39
  #endif
Packit Service 360c39
#endif
Packit Service 360c39
Packit Service 360c39
struct gfs2_buffer_head *bget(struct gfs2_sbd *sdp, uint64_t num)
Packit Service 360c39
{
Packit Service 360c39
	struct gfs2_buffer_head *bh;
Packit Service 360c39
Packit Service 360c39
	bh = calloc(1, sizeof(struct gfs2_buffer_head) + sdp->bsize);
Packit Service 360c39
	if (bh == NULL)
Packit Service 360c39
		return NULL;
Packit Service 360c39
Packit Service 360c39
	bh->b_blocknr = num;
Packit Service 360c39
	bh->sdp = sdp;
Packit Service 360c39
	bh->iov.iov_base = (char *)bh + sizeof(struct gfs2_buffer_head);
Packit Service 360c39
	bh->iov.iov_len = sdp->bsize;
Packit Service 360c39
Packit Service 360c39
	return bh;
Packit Service 360c39
}
Packit Service 360c39
Packit Service 360c39
int __breadm(struct gfs2_sbd *sdp, struct gfs2_buffer_head **bhs, size_t n,
Packit Service 360c39
	     uint64_t block, int line, const char *caller)
Packit Service 360c39
{
Packit Service 360c39
	size_t v = (n < IOV_MAX) ? n : IOV_MAX;
Packit Service 360c39
	struct iovec *iov = alloca(v * sizeof(struct iovec));
Packit Service 360c39
	struct iovec *iovbase = iov;
Packit Service 360c39
	size_t i = 0;
Packit Service 360c39
Packit Service 360c39
	while (i < n) {
Packit Service 360c39
		int j;
Packit Service 360c39
		ssize_t ret;
Packit Service 360c39
		ssize_t size = 0;
Packit Service 360c39
Packit Service 360c39
		for (j = 0; (i + j < n) && (j < IOV_MAX); j++) {
Packit Service 360c39
			bhs[i + j] = bget(sdp, block + i + j);
Packit Service 360c39
			if (bhs[i + j] == NULL)
Packit Service 360c39
				return -1;
Packit Service 360c39
			iov[j] = bhs[i + j]->iov;
Packit Service 360c39
			size += bhs[i + j]->iov.iov_len;
Packit Service 360c39
		}
Packit Service 360c39
Packit Service 360c39
		ret = preadv(sdp->device_fd, iovbase, j, (block + i) * sdp->bsize);
Packit Service 360c39
		if (ret != size) {
Packit Service 360c39
			fprintf(stderr, "bad read: %s from %s:%d: block %llu (0x%llx) "
Packit Service 360c39
					"count: %d size: %zd ret: %zd\n", strerror(errno),
Packit Service 360c39
					caller, line, (unsigned long long)block,
Packit Service 360c39
					(unsigned long long)block, j, size, ret);
Packit Service 360c39
			exit(-1);
Packit Service 360c39
		}
Packit Service 360c39
		i += j;
Packit Service 360c39
	}
Packit Service 360c39
	return 0;
Packit Service 360c39
}
Packit Service 360c39
Packit Service 360c39
struct gfs2_buffer_head *__bread(struct gfs2_sbd *sdp, uint64_t num, int line,
Packit Service 360c39
				 const char *caller)
Packit Service 360c39
{
Packit Service 360c39
	struct gfs2_buffer_head *bh;
Packit Service 360c39
	int ret;
Packit Service 360c39
Packit Service 360c39
	ret = __breadm(sdp, &bh, 1, num, line, caller);
Packit Service 360c39
	if (ret >= 0)
Packit Service 360c39
		return bh;
Packit Service 360c39
	return NULL;
Packit Service 360c39
}
Packit Service 360c39
Packit Service 360c39
int bwrite(struct gfs2_buffer_head *bh)
Packit Service 360c39
{
Packit Service 360c39
	struct gfs2_sbd *sdp = bh->sdp;
Packit Service 360c39
Packit Service 360c39
	if (pwritev(sdp->device_fd, &bh->iov, 1, bh->b_blocknr * sdp->bsize) != bh->iov.iov_len)
Packit Service 360c39
		return -1;
Packit Service 360c39
	bh->b_modified = 0;
Packit Service 360c39
	return 0;
Packit Service 360c39
}
Packit Service 360c39
Packit Service 360c39
int brelse(struct gfs2_buffer_head *bh)
Packit Service 360c39
{
Packit Service 360c39
	int error = 0;
Packit Service 360c39
Packit Service 360c39
	if (bh->b_blocknr == -1)
Packit Service 360c39
		printf("Double free!\n");
Packit Service 360c39
	if (bh->b_modified)
Packit Service 360c39
		error = bwrite(bh);
Packit Service 360c39
	bh->b_blocknr = -1;
Packit Service 360c39
	if (bh->b_altlist.next && !osi_list_empty(&bh->b_altlist))
Packit Service 360c39
		osi_list_del(&bh->b_altlist);
Packit Service 360c39
	free(bh);
Packit Service 360c39
	return error;
Packit Service 360c39
}
Packit Service 360c39
Packit Service 360c39
uint32_t lgfs2_get_block_type(const struct gfs2_buffer_head *lbh)
Packit Service 360c39
{
Packit Service 360c39
	const struct gfs2_meta_header *mh = lbh->iov.iov_base;
Packit Service 360c39
Packit Service 360c39
	if (be32_to_cpu(mh->mh_magic) == GFS2_MAGIC)
Packit Service 360c39
		return be32_to_cpu(mh->mh_type);
Packit Service 360c39
Packit Service 360c39
	return 0;
Packit Service 360c39
}