/* $NetBSD$ */ /*- * Copyright (c) 2016 The NetBSD Foundation, Inc. * All rights reserved. * * This code is derived from software contributed to The NetBSD Foundation * by Taylor R. Campbell. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #ifndef _SYS_PSREF_H #define _SYS_PSREF_H #include #include /* * PSREF_DEBUG * * If nonzero, enable debugging of psrefs. WARNING: This changes * the ABI by adding extra fields to struct psref_target and * struct psref, which are exposed to callers and embedded in * other structures. */ #ifndef PSREF_DEBUG #define PSREF_DEBUG 0 #endif struct cpu_info; struct lwp; struct psref; struct psref_class; struct psref_target; /* * struct psref_target * * Bookkeeping for an object to which users can acquire passive * references. This is compact so that it can easily be embedded * into many multitudes of objects, e.g. IP packet flows. * * prt_draining is false on initialization, and may be written * only once, to make it true, when someone with exclusive access * to it is about to drain the target. */ struct psref_target { bool prt_draining; #if PSREF_DEBUG struct psref_class *prt_class; #endif }; /* * struct psref * * Bookkeeping for a single passive reference. There should only * be a few of these per CPU in the system at once, no matter how * many targets are stored, so these are a bit larger than struct * psref_target. The contents of struct psref may be read and * written only on the local CPU. */ struct psref { LIST_ENTRY(psref) psref_entry; struct psref_target *psref_target; #if PSREF_DEBUG struct lwp *psref_lwp; struct cpu_info *psref_cpu; #endif }; struct psref_class * psref_class_create(const char *, int); void psref_class_destroy(struct psref_class *); void psref_target_init(struct psref_target *, struct psref_class *); void psref_target_destroy(struct psref_target *, struct psref_class *); int psref_acquire(struct psref *, struct psref_target *, struct psref_class *); void psref_release(struct psref *, struct psref_target *, struct psref_class *); void psref_target_drain(struct psref_target *, struct psref_class *); #endif /* _SYS_PSREF_H */