@@ -0,0 +1,255 @@
/*
* GPL HEADER START
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 only,
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License version 2 for more details (a copy is included
* in the LICENSE file that accompanied this code).
*
* You should have received a copy of the GNU General Public License
* version 2 along with this program; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 021110-1307, USA
*
* GPL HEADER END
*/
/* Copyright (c) 2010, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2011, 2012, Whamcloud, Inc.
*/
/*
* This file is part of Lustre, http://www.lustre.org/
* Lustre is a trademark of Sun Microsystems, Inc.
*
* Author: liang@whamcloud.com
*/

#ifndef EXPORT_SYMTAB
# define EXPORT_SYMTAB
#endif
#define DEBUG_SUBSYSTEM S_LNET

#include <libcfs/libcfs.h>

#ifdef __KERNEL__

/** destroy cpu-partition lock, see libcfs_private.h for more detail */
void
cfs_percpt_lock_free(struct cfs_percpt_lock *pcl)
{
LASSERT(pcl->pcl_locks != NULL);
LASSERT(!pcl->pcl_locked);

cfs_percpt_free(pcl->pcl_locks);
LIBCFS_FREE(pcl, sizeof(*pcl));
}
CFS_EXPORT_SYMBOL(cfs_percpt_lock_free);

/**
* create cpu-partition lock, see libcfs_private.h for more detail.
*
* cpu-partition lock is designed for large-scale SMP system, so we need to
* reduce cacheline conflict as possible as we can, that's the
* reason we always allocate cacheline-aligned memory block.
*/
struct cfs_percpt_lock *
cfs_percpt_lock_alloc(struct cfs_cpt_table *cptab)
{
struct cfs_percpt_lock *pcl;
cfs_spinlock_t *lock;
int i;

/* NB: cptab can be NULL, pcl will be for HW CPUs on that case */
LIBCFS_ALLOC(pcl, sizeof(*pcl));
if (pcl == NULL)
return NULL;

pcl->pcl_cptab = cptab;
pcl->pcl_locks = cfs_percpt_alloc(cptab, sizeof(*lock));
if (pcl->pcl_locks == NULL) {
LIBCFS_FREE(pcl, sizeof(*pcl));
return NULL;
}

cfs_percpt_for_each(lock, i, pcl->pcl_locks)
cfs_spin_lock_init(lock);

return pcl;
}
CFS_EXPORT_SYMBOL(cfs_percpt_lock_alloc);

/**
* lock a CPU partition
*
* \a index != CFS_PERCPT_LOCK_EX
* hold private lock indexed by \a index
*
* \a index == CFS_PERCPT_LOCK_EX
* exclusively lock @pcl and nobody can take private lock
*/
void
cfs_percpt_lock(struct cfs_percpt_lock *pcl, int index)
{
int ncpt = cfs_cpt_number(pcl->pcl_cptab);
int i;

LASSERT(index >= CFS_PERCPT_LOCK_EX && index < ncpt);

if (ncpt == 1) {
index = 0;
} else { /* serialize with exclusive lock */
while (pcl->pcl_locked)
cpu_relax();
}

if (likely(index != CFS_PERCPT_LOCK_EX)) {
cfs_spin_lock(pcl->pcl_locks[index]);
return;
}

/* exclusive lock request */
for (i = 0; i < ncpt; i++) {
cfs_spin_lock(pcl->pcl_locks[i]);
if (i == 0) {
LASSERT(!pcl->pcl_locked);
/* nobody should take private lock after this
* so I wouldn't starve for too long time */
pcl->pcl_locked = 1;
}
}
}
CFS_EXPORT_SYMBOL(cfs_percpt_lock);

/** unlock a CPU partition */
void
cfs_percpt_unlock(struct cfs_percpt_lock *pcl, int index)
{
int ncpt = cfs_cpt_number(pcl->pcl_cptab);
int i;

index = ncpt == 1 ? 0 : index;

if (likely(index != CFS_PERCPT_LOCK_EX)) {
cfs_spin_unlock(pcl->pcl_locks[index]);
return;
}

for (i = ncpt - 1; i >= 0; i--) {
if (i == 0) {
LASSERT(pcl->pcl_locked);
pcl->pcl_locked = 0;
}
cfs_spin_unlock(pcl->pcl_locks[i]);
}
}
CFS_EXPORT_SYMBOL(cfs_percpt_unlock);

#else /* !__KERNEL__ */
# ifdef HAVE_LIBPTHREAD

struct cfs_percpt_lock *
cfs_percpt_lock_alloc(struct cfs_cpt_table *cptab)
{
struct cfs_percpt_lock *pcl;

CFS_ALLOC_PTR(pcl);
if (pcl != NULL)
pthread_mutex_init(&pcl->pcl_mutex, NULL);

return pcl;
}

void
cfs_percpt_lock_free(struct cfs_percpt_lock *pcl)
{
pthread_mutex_destroy(&pcl->pcl_mutex);
CFS_FREE_PTR(pcl);
}

void
cfs_percpt_lock(struct cfs_percpt_lock *pcl, int lock)
{
pthread_mutex_lock(&(pcl)->pcl_mutex);
}

void
cfs_percpt_unlock(struct cfs_percpt_lock *pcl, int lock)
{
pthread_mutex_unlock(&(pcl)->pcl_mutex);
}

# else /* !HAVE_LIBPTHREAD */

struct cfs_percpt_lock *
cfs_percpt_lock_alloc(struct cfs_cpt_table *cptab)
{
return (struct cfs_percpt_lock *)CFS_PERCPT_LOCK_MAGIC;
}

void
cfs_percpt_lock_free(struct cfs_percpt_lock *pcl)
{
LASSERT(pcl == (struct cfs_percpt_lock *)CFS_PERCPT_LOCK_MAGIC);
}

void
cfs_percpt_lock(struct cfs_percpt_lock *pcl, int index)
{
LASSERT(pcl == (struct cfs_percpt_lock *)CFS_PERCPT_LOCK_MAGIC);
}

void
cfs_percpt_unlock(struct cfs_percpt_lock *pcl, int index)
{
LASSERT(pcl == (struct cfs_percpt_lock *)CFS_PERCPT_LOCK_MAGIC);
}

# endif /* HAVE_LIBPTHREAD */
#endif /* __KERNEL__ */

/** free cpu-partition refcount */
void
cfs_percpt_atomic_free(cfs_atomic_t **refs)
{
cfs_percpt_free(refs);
}
CFS_EXPORT_SYMBOL(cfs_percpt_atomic_free);

/** allocate cpu-partition refcount with initial value @init_val */
cfs_atomic_t **
cfs_percpt_atomic_alloc(struct cfs_cpt_table *cptab, int init_val)
{
cfs_atomic_t **refs;
cfs_atomic_t *ref;
int i;

refs = cfs_percpt_alloc(cptab, sizeof(*ref));
if (refs == NULL)
return NULL;

cfs_percpt_for_each(ref, i, refs)
cfs_atomic_set(ref, init_val);
return refs;
}
CFS_EXPORT_SYMBOL(cfs_percpt_atomic_alloc);

/** return sum of cpu-partition refs */
int
cfs_percpt_atomic_summary(cfs_atomic_t **refs)
{
cfs_atomic_t *ref;
int i;
int val = 0;

cfs_percpt_for_each(ref, i, refs)
val += cfs_atomic_read(ref);

return val;
}
CFS_EXPORT_SYMBOL(cfs_percpt_atomic_summary);
@@ -0,0 +1,205 @@
/*
* GPL HEADER START
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 only,
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License version 2 for more details (a copy is included
* in the LICENSE file that accompanied this code).
*
* You should have received a copy of the GNU General Public License
* version 2 along with this program; if not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 021110-1307, USA
*
* GPL HEADER END
*/
/*
* Copyright (c) 2010, Oracle and/or its affiliates. All rights reserved.
* Copyright (c) 2011, 2012, Whamcloud, Inc.
*/
/*
* This file is part of Lustre, http://www.lustre.org/
* Lustre is a trademark of Sun Microsystems, Inc.
*
* Author: liang@whamcloud.com
*/

#ifndef EXPORT_SYMTAB
# define EXPORT_SYMTAB
#endif
#define DEBUG_SUBSYSTEM S_LNET

#include <libcfs/libcfs.h>

struct cfs_var_array {
unsigned int va_count; /* # of buffers */
unsigned int va_size; /* size of each var */
struct cfs_cpt_table *va_cptab; /* cpu partition table */
void *va_ptrs[0]; /* buffer addresses */
};

/*
* free per-cpu data, see more detail in cfs_percpt_free
*/
void
cfs_percpt_free(void *vars)
{
struct cfs_var_array *arr;
int i;

arr = container_of(vars, struct cfs_var_array, va_ptrs[0]);

for (i = 0; i < arr->va_count; i++) {
if (arr->va_ptrs[i] != NULL)
LIBCFS_FREE(arr->va_ptrs[i], arr->va_size);
}

LIBCFS_FREE(arr, offsetof(struct cfs_var_array,
va_ptrs[arr->va_count]));
}
EXPORT_SYMBOL(cfs_percpt_free);

/*
* allocate per cpu-partition variables, returned value is an array of pointers,
* variable can be indexed by CPU partition ID, i.e:
*
* arr = cfs_percpt_alloc(cfs_cpu_pt, size);
* then caller can access memory block for CPU 0 by arr[0],
* memory block for CPU 1 by arr[1]...
* memory block for CPU N by arr[N]...
*
* cacheline aligned.
*/
void *
cfs_percpt_alloc(struct cfs_cpt_table *cptab, unsigned int size)
{
struct cfs_var_array *arr;
int count;
int i;

count = cfs_cpt_number(cptab);

LIBCFS_ALLOC(arr, offsetof(struct cfs_var_array, va_ptrs[count]));
if (arr == NULL)
return NULL;

arr->va_size = size = CFS_L1_CACHE_ALIGN(size);
arr->va_count = count;
arr->va_cptab = cptab;

for (i = 0; i < count; i++) {
LIBCFS_CPT_ALLOC(arr->va_ptrs[i], cptab, i, size);
if (arr->va_ptrs[i] == NULL) {
cfs_percpt_free((void *)&arr->va_ptrs[0]);
return NULL;
}
}

return (void *)&arr->va_ptrs[0];
}
EXPORT_SYMBOL(cfs_percpt_alloc);

/*
* return number of CPUs (or number of elements in per-cpu data)
* according to cptab of @vars
*/
int
cfs_percpt_number(void *vars)
{
struct cfs_var_array *arr;

arr = container_of(vars, struct cfs_var_array, va_ptrs[0]);

return arr->va_count;
}
EXPORT_SYMBOL(cfs_percpt_number);

/*
* return memory block shadowed from current CPU
*/
void *
cfs_percpt_current(void *vars)
{
struct cfs_var_array *arr;
int cpt;

arr = container_of(vars, struct cfs_var_array, va_ptrs[0]);
cpt = cfs_cpt_current(arr->va_cptab, 0);
if (cpt < 0)
return NULL;

return arr->va_ptrs[cpt];
}
EXPORT_SYMBOL(cfs_percpt_current);

void *
cfs_percpt_index(void *vars, int idx)
{
struct cfs_var_array *arr;

arr = container_of(vars, struct cfs_var_array, va_ptrs[0]);

LASSERT(idx >= 0 && idx < arr->va_count);
return arr->va_ptrs[idx];
}
EXPORT_SYMBOL(cfs_percpt_index);

/*
* free variable array, see more detail in cfs_array_alloc
*/
void
cfs_array_free(void *vars)
{
struct cfs_var_array *arr;
int i;

arr = container_of(vars, struct cfs_var_array, va_ptrs[0]);

for (i = 0; i < arr->va_count; i++) {
if (arr->va_ptrs[i] == NULL)
continue;

LIBCFS_FREE(arr->va_ptrs[i], arr->va_size);
}
LIBCFS_FREE(arr, offsetof(struct cfs_var_array,
va_ptrs[arr->va_count]));
}
EXPORT_SYMBOL(cfs_array_free);

/*
* allocate a variable array, returned value is an array of pointers.
* Caller can specify length of array by @count, @size is size of each
* memory block in array.
*/
void *
cfs_array_alloc(int count, unsigned int size)
{
struct cfs_var_array *arr;
int i;

LIBCFS_ALLOC(arr, offsetof(struct cfs_var_array, va_ptrs[count]));
if (arr == NULL)
return NULL;

arr->va_count = count;
arr->va_size = size;

for (i = 0; i < count; i++) {
LIBCFS_ALLOC(arr->va_ptrs[i], size);

if (arr->va_ptrs[i] == NULL) {
cfs_array_free((void *)&arr->va_ptrs[0]);
return NULL;
}
}

return (void *)&arr->va_ptrs[0];
}
EXPORT_SYMBOL(cfs_array_alloc);