exec_list: Add macros to get ptr to structure containing a node
authorIan Romanick <ian.d.romanick@intel.com>
Mon, 10 May 2010 18:16:24 +0000 (11:16 -0700)
committerIan Romanick <ian.d.romanick@intel.com>
Mon, 10 May 2010 18:16:24 +0000 (11:16 -0700)
This has some ugly hackery to work-around C++ fail.  I have
emperically determined that it works in all the cases that matter.

list.h

diff --git a/list.h b/list.h
index afa32f1..2694174 100644 (file)
--- a/list.h
+++ b/list.h
@@ -64,6 +64,9 @@
 #ifndef LIST_CONTAINER_H
 #define LIST_CONTAINER_H
 
+#ifndef __cplusplus
+#include <stddef.h>
+#endif
 #include <assert.h>
 
 struct exec_node {
@@ -140,6 +143,30 @@ struct exec_node {
 #endif
 };
 
+
+#ifdef __cplusplus
+/* This macro will not work correctly if `t' uses virtual inheritance.  If you
+ * are using virtual inheritance, you deserve a slow and painful death.  Enjoy!
+ */
+#define exec_list_offsetof(t, f, p) \
+   (((char *) &((t *) p)->f) - ((char *) p))
+#else
+#define exec_list_offsetof(t, f, p) offsetof(t, f)
+#endif
+
+/**
+ * Get a pointer to the structure containing an exec_node
+ *
+ * Given a pointer to an \c exec_node embedded in a structure, get a pointer to
+ * the containing structure.
+ *
+ * \param type  Base type of the structure containing the node
+ * \param node  Pointer to the \c exec_node
+ * \param field Name of the field in \c type that is the embedded \c exec_node
+ */
+#define exec_node_data(type, node, field) \
+   ((type *) (((char *) node) - exec_list_offsetof(type, field, node)))
+
 #ifdef __cplusplus
 struct exec_node;