diff options
| author | Michael Brown | 2007-04-30 01:53:39 +0200 |
|---|---|---|
| committer | Michael Brown | 2007-04-30 01:53:39 +0200 |
| commit | d40761d7258cd509d698c210fc6c478bd7788c6b (patch) | |
| tree | c7f419e1e189936d706a53c26fc9f236e0118b6d /src/core | |
| parent | Skeleton job control interface (diff) | |
| download | ipxe-d40761d7258cd509d698c210fc6c478bd7788c6b.tar.gz ipxe-d40761d7258cd509d698c210fc6c478bd7788c6b.tar.xz ipxe-d40761d7258cd509d698c210fc6c478bd7788c6b.zip | |
Add (and use) generic reference counter, to improve signal:noise ratio
in code defining reference-counted objects.
Diffstat (limited to 'src/core')
| -rw-r--r-- | src/core/interface.c | 43 | ||||
| -rw-r--r-- | src/core/job.c | 2 | ||||
| -rw-r--r-- | src/core/refcnt.c | 73 | ||||
| -rw-r--r-- | src/core/xfer.c | 2 |
4 files changed, 78 insertions, 42 deletions
diff --git a/src/core/interface.c b/src/core/interface.c index 15df49d3c..8498425cd 100644 --- a/src/core/interface.c +++ b/src/core/interface.c @@ -16,7 +16,6 @@ * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ -#include <assert.h> #include <gpxe/interface.h> /** @file @@ -26,30 +25,6 @@ */ /** - * Obtain reference to interface - * - * @v intf Interface - * @ret intf Interface - * - * Increases the reference count on the interface. - */ -static struct interface * intf_get ( struct interface *intf ) { - intf->refcnt ( intf, +1 ); - return intf; -} - -/** - * Drop reference to interface - * - * @v intf Interface - * - * Decreases the reference count on the interface. - */ -static void intf_put ( struct interface *intf ) { - intf->refcnt ( intf, -1 ); -} - -/** * Plug an interface into a new destination interface * * @v intf Interface @@ -63,19 +38,7 @@ static void intf_put ( struct interface *intf ) { * interface into a null interface. */ void plug ( struct interface *intf, struct interface *dest ) { - intf_put ( intf->dest ); - intf->dest = intf_get ( dest ); -} - -/** - * Null update reference count - * - * @v intf Interface - * @v delta Change to apply to reference count - * - * Use this as the refcnt() method for an interface that does not need - * to support reference counting. - */ -void null_refcnt ( struct interface *intf __unused, int delta __unused ) { - /* Do nothing */ + ref_put ( intf->refcnt ); + ref_get ( dest->refcnt ); + intf->dest = dest; } diff --git a/src/core/job.c b/src/core/job.c index fc60f4ece..d6ec27bc0 100644 --- a/src/core/job.c +++ b/src/core/job.c @@ -71,7 +71,7 @@ struct job_interface_operations null_job_ops = { struct job_interface null_job = { .intf = { .dest = &null_job.intf, - .refcnt = null_refcnt, + .refcnt = NULL, }, .op = &null_job_ops, }; diff --git a/src/core/refcnt.c b/src/core/refcnt.c new file mode 100644 index 000000000..4efd9d715 --- /dev/null +++ b/src/core/refcnt.c @@ -0,0 +1,73 @@ +/* + * Copyright (C) 2007 Michael Brown <mbrown@fensystems.co.uk>. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or any later version. + * + * 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 for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + */ + +#include <stdlib.h> +#include <gpxe/refcnt.h> + +/** @file + * + * Reference counting + * + */ + +/** + * Increment reference count + * + * @v refcnt Reference counter, or NULL + * + * If @c refcnt is NULL, no action is taken. + */ +void ref_get ( struct refcnt *refcnt ) { + + if ( ! refcnt ) + return; + + refcnt->refcnt++; + + DBGC ( refcnt, "REFCNT %p incremented to %d\n", + refcnt, refcnt->refcnt ); +} + +/** + * Decrement reference count + * + * @v refcnt Reference counter, or NULL + * + * If the reference count decreases below zero, the object's free() + * method will be called. + * + * If @c refcnt is NULL, no action is taken. + */ +void ref_put ( struct refcnt *refcnt ) { + + if ( ! refcnt ) + return; + + refcnt->refcnt--; + DBGC ( refcnt, "REFCNT %p decremented to %d\n", + refcnt, refcnt->refcnt ); + + if ( refcnt->refcnt >= 0 ) + return; + + if ( refcnt->free ) { + refcnt->free ( refcnt ); + } else { + free ( refcnt ); + } +} diff --git a/src/core/xfer.c b/src/core/xfer.c index d34808166..f927269ea 100644 --- a/src/core/xfer.c +++ b/src/core/xfer.c @@ -230,7 +230,7 @@ struct xfer_interface_operations null_xfer_ops = { struct xfer_interface null_xfer = { .intf = { .dest = &null_xfer.intf, - .refcnt = null_refcnt, + .refcnt = NULL, }, .op = &null_xfer_ops, }; |
