From: Bill Fischofer <[email protected]>

Modify pool create API to be able to specify number of buffers directly.

Signed-off-by: Bill Fischofer <[email protected]>
Signed-off-by: Taras Kondratiuk <[email protected]>
---
 example/generator/odp_generator.c                  |  19 +-
 example/ipsec/odp_ipsec.c                          |  57 +-
 example/l2fwd/odp_l2fwd.c                          |  19 +-
 example/odp_example/odp_example.c                  |  18 +-
 example/packet/odp_pktio.c                         |  20 +-
 example/timer/odp_timer_test.c                     |  13 +-
 .../linux-generic/include/api/odp_buffer_pool.h    |  47 +-
 .../linux-generic/include/api/odp_platform_types.h |   3 +-
 .../linux-generic/include/odp_buffer_inlines.h     | 150 ++++
 .../linux-generic/include/odp_buffer_internal.h    | 152 ++--
 .../include/odp_buffer_pool_internal.h             | 346 ++++++++--
 platform/linux-generic/include/odp_internal.h      |   2 +
 .../linux-generic/include/odp_packet_internal.h    |  50 +-
 .../linux-generic/include/odp_timer_internal.h     |  11 +-
 platform/linux-generic/odp_buffer.c                |  33 +-
 platform/linux-generic/odp_buffer_pool.c           | 762 ++++++++++-----------
 platform/linux-generic/odp_linux.c                 |   4 +-
 platform/linux-generic/odp_packet.c                |  41 +-
 platform/linux-generic/odp_queue.c                 |   1 +
 platform/linux-generic/odp_schedule.c              |  20 +-
 platform/linux-generic/odp_timer.c                 |   3 +-
 test/api_test/odp_timer_ping.c                     |  19 +-
 test/validation/odp_crypto.c                       |  43 +-
 test/validation/odp_queue.c                        |  19 +-
 test/validation/odp_schedule.c                     |  18 +-
 25 files changed, 1112 insertions(+), 758 deletions(-)
 create mode 100644 platform/linux-generic/include/odp_buffer_inlines.h

diff --git a/example/generator/odp_generator.c 
b/example/generator/odp_generator.c
index 8c227bc..a67910e 100644
--- a/example/generator/odp_generator.c
+++ b/example/generator/odp_generator.c
@@ -522,11 +522,11 @@ int main(int argc, char *argv[])
        odph_linux_pthread_t thread_tbl[MAX_WORKERS];
        odp_buffer_pool_t pool;
        int num_workers;
-       void *pool_base;
        int i;
        int first_core;
        int core_count;
        odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
        /* Init ODP before calling anything else */
        if (odp_init_global(NULL, NULL)) {
@@ -589,20 +589,13 @@ int main(int argc, char *argv[])
        printf("First core:         %i\n\n", first_core);
 
        /* Create packet pool */
-       shm = odp_shm_reserve("shm_packet_pool",
-                             SHM_PKT_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
-       pool_base = odp_shm_addr(shm);
+       params.buf_size  = SHM_PKT_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_PKT_POOL_SIZE/SHM_PKT_POOL_BUF_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_PACKET;
 
-       if (pool_base == NULL) {
-               EXAMPLE_ERR("Error: packet pool mem alloc failed.\n");
-               exit(EXIT_FAILURE);
-       }
+       pool = odp_buffer_pool_create("packet_pool", ODP_SHM_NULL, &params);
 
-       pool = odp_buffer_pool_create("packet_pool", pool_base,
-                                     SHM_PKT_POOL_SIZE,
-                                     SHM_PKT_POOL_BUF_SIZE,
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_PACKET);
        if (pool == ODP_BUFFER_POOL_INVALID) {
                EXAMPLE_ERR("Error: packet pool create failed.\n");
                exit(EXIT_FAILURE);
diff --git a/example/ipsec/odp_ipsec.c b/example/ipsec/odp_ipsec.c
index 538f8dc..4b5a21c 100644
--- a/example/ipsec/odp_ipsec.c
+++ b/example/ipsec/odp_ipsec.c
@@ -367,8 +367,7 @@ static
 void ipsec_init_pre(void)
 {
        odp_queue_param_t qparam;
-       void *pool_base;
-       odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
        /*
         * Create queues
@@ -401,16 +400,12 @@ void ipsec_init_pre(void)
        }
 
        /* Create output buffer pool */
-       shm = odp_shm_reserve("shm_out_pool",
-                             SHM_OUT_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
-
-       pool_base = odp_shm_addr(shm);
+       params.buf_size  = SHM_OUT_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_PKT_POOL_BUF_COUNT;
+       params.buf_type  = ODP_BUFFER_TYPE_PACKET;
 
-       out_pool = odp_buffer_pool_create("out_pool", pool_base,
-                                         SHM_OUT_POOL_SIZE,
-                                         SHM_OUT_POOL_BUF_SIZE,
-                                         ODP_CACHE_LINE_SIZE,
-                                         ODP_BUFFER_TYPE_PACKET);
+       out_pool = odp_buffer_pool_create("out_pool", ODP_SHM_NULL, &params);
 
        if (ODP_BUFFER_POOL_INVALID == out_pool) {
                EXAMPLE_ERR("Error: message pool create failed.\n");
@@ -1176,12 +1171,12 @@ main(int argc, char *argv[])
 {
        odph_linux_pthread_t thread_tbl[MAX_WORKERS];
        int num_workers;
-       void *pool_base;
        int i;
        int first_core;
        int core_count;
        int stream_count;
        odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
        /* Init ODP before calling anything else */
        if (odp_init_global(NULL, NULL)) {
@@ -1241,42 +1236,28 @@ main(int argc, char *argv[])
        printf("First core:         %i\n\n", first_core);
 
        /* Create packet buffer pool */
-       shm = odp_shm_reserve("shm_packet_pool",
-                             SHM_PKT_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
+       params.buf_size  = SHM_PKT_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_PKT_POOL_BUF_COUNT;
+       params.buf_type  = ODP_BUFFER_TYPE_PACKET;
 
-       pool_base = odp_shm_addr(shm);
-
-       if (NULL == pool_base) {
-               EXAMPLE_ERR("Error: packet pool mem alloc failed.\n");
-               exit(EXIT_FAILURE);
-       }
+       pkt_pool = odp_buffer_pool_create("packet_pool", ODP_SHM_NULL,
+                                         &params);
 
-       pkt_pool = odp_buffer_pool_create("packet_pool", pool_base,
-                                         SHM_PKT_POOL_SIZE,
-                                         SHM_PKT_POOL_BUF_SIZE,
-                                         ODP_CACHE_LINE_SIZE,
-                                         ODP_BUFFER_TYPE_PACKET);
        if (ODP_BUFFER_POOL_INVALID == pkt_pool) {
                EXAMPLE_ERR("Error: packet pool create failed.\n");
                exit(EXIT_FAILURE);
        }
 
        /* Create context buffer pool */
-       shm = odp_shm_reserve("shm_ctx_pool",
-                             SHM_CTX_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
-
-       pool_base = odp_shm_addr(shm);
+       params.buf_size  = SHM_CTX_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_CTX_POOL_BUF_COUNT;
+       params.buf_type  = ODP_BUFFER_TYPE_RAW;
 
-       if (NULL == pool_base) {
-               EXAMPLE_ERR("Error: context pool mem alloc failed.\n");
-               exit(EXIT_FAILURE);
-       }
+       ctx_pool = odp_buffer_pool_create("ctx_pool", ODP_SHM_NULL,
+                                         &params);
 
-       ctx_pool = odp_buffer_pool_create("ctx_pool", pool_base,
-                                         SHM_CTX_POOL_SIZE,
-                                         SHM_CTX_POOL_BUF_SIZE,
-                                         ODP_CACHE_LINE_SIZE,
-                                         ODP_BUFFER_TYPE_RAW);
        if (ODP_BUFFER_POOL_INVALID == ctx_pool) {
                EXAMPLE_ERR("Error: context pool create failed.\n");
                exit(EXIT_FAILURE);
diff --git a/example/l2fwd/odp_l2fwd.c b/example/l2fwd/odp_l2fwd.c
index ebac8c5..3c1fd6a 100644
--- a/example/l2fwd/odp_l2fwd.c
+++ b/example/l2fwd/odp_l2fwd.c
@@ -314,12 +314,12 @@ int main(int argc, char *argv[])
 {
        odph_linux_pthread_t thread_tbl[MAX_WORKERS];
        odp_buffer_pool_t pool;
-       void *pool_base;
        int i;
        int first_core;
        int core_count;
        odp_pktio_t pktio;
        odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
        /* Init ODP before calling anything else */
        if (odp_init_global(NULL, NULL)) {
@@ -383,20 +383,13 @@ int main(int argc, char *argv[])
        printf("First core:         %i\n\n", first_core);
 
        /* Create packet pool */
-       shm = odp_shm_reserve("shm_packet_pool",
-                             SHM_PKT_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
-       pool_base = odp_shm_addr(shm);
+       params.buf_size  = SHM_PKT_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_PKT_POOL_SIZE/SHM_PKT_POOL_BUF_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_PACKET;
 
-       if (pool_base == NULL) {
-               EXAMPLE_ERR("Error: packet pool mem alloc failed.\n");
-               exit(EXIT_FAILURE);
-       }
+       pool = odp_buffer_pool_create("packet pool", ODP_SHM_NULL, &params);
 
-       pool = odp_buffer_pool_create("packet_pool", pool_base,
-                                     SHM_PKT_POOL_SIZE,
-                                     SHM_PKT_POOL_BUF_SIZE,
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_PACKET);
        if (pool == ODP_BUFFER_POOL_INVALID) {
                EXAMPLE_ERR("Error: packet pool create failed.\n");
                exit(EXIT_FAILURE);
diff --git a/example/odp_example/odp_example.c 
b/example/odp_example/odp_example.c
index 96a2912..8373f12 100644
--- a/example/odp_example/odp_example.c
+++ b/example/odp_example/odp_example.c
@@ -954,13 +954,13 @@ int main(int argc, char *argv[])
        test_args_t args;
        int num_workers;
        odp_buffer_pool_t pool;
-       void *pool_base;
        odp_queue_t queue;
        int i, j;
        int prios;
        int first_core;
        odp_shm_t shm;
        test_globals_t *globals;
+       odp_buffer_pool_param_t params;
 
        printf("\nODP example starts\n\n");
 
@@ -1042,19 +1042,13 @@ int main(int argc, char *argv[])
        /*
         * Create message pool
         */
-       shm = odp_shm_reserve("msg_pool",
-                             MSG_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
 
-       pool_base = odp_shm_addr(shm);
+       params.buf_size  = sizeof(test_message_t);
+       params.buf_align = 0;
+       params.num_bufs  = MSG_POOL_SIZE/sizeof(test_message_t);
+       params.buf_type  = ODP_BUFFER_TYPE_RAW;
 
-       if (pool_base == NULL) {
-               EXAMPLE_ERR("Shared memory reserve failed.\n");
-               return -1;
-       }
-
-       pool = odp_buffer_pool_create("msg_pool", pool_base, MSG_POOL_SIZE,
-                                     sizeof(test_message_t),
-                                     ODP_CACHE_LINE_SIZE, ODP_BUFFER_TYPE_RAW);
+       pool = odp_buffer_pool_create("msg_pool", ODP_SHM_NULL, &params);
 
        if (pool == ODP_BUFFER_POOL_INVALID) {
                EXAMPLE_ERR("Pool create failed.\n");
diff --git a/example/packet/odp_pktio.c b/example/packet/odp_pktio.c
index 607ecbf..5ac8398 100644
--- a/example/packet/odp_pktio.c
+++ b/example/packet/odp_pktio.c
@@ -331,11 +331,10 @@ int main(int argc, char *argv[])
        odph_linux_pthread_t thread_tbl[MAX_WORKERS];
        odp_buffer_pool_t pool;
        int num_workers;
-       void *pool_base;
        int i;
        int first_core;
        int core_count;
-       odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
        args = calloc(1, sizeof(args_t));
        if (args == NULL) {
@@ -384,20 +383,13 @@ int main(int argc, char *argv[])
        printf("First core:         %i\n\n", first_core);
 
        /* Create packet pool */
-       shm = odp_shm_reserve("shm_packet_pool",
-                             SHM_PKT_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
-       pool_base = odp_shm_addr(shm);
+       params.buf_size  = SHM_PKT_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_PKT_POOL_SIZE/SHM_PKT_POOL_BUF_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_PACKET;
 
-       if (pool_base == NULL) {
-               EXAMPLE_ERR("Error: packet pool mem alloc failed.\n");
-               exit(EXIT_FAILURE);
-       }
+       pool = odp_buffer_pool_create("packet_pool", ODP_SHM_NULL, &params);
 
-       pool = odp_buffer_pool_create("packet_pool", pool_base,
-                                     SHM_PKT_POOL_SIZE,
-                                     SHM_PKT_POOL_BUF_SIZE,
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_PACKET);
        if (pool == ODP_BUFFER_POOL_INVALID) {
                EXAMPLE_ERR("Error: packet pool create failed.\n");
                exit(EXIT_FAILURE);
diff --git a/example/timer/odp_timer_test.c b/example/timer/odp_timer_test.c
index 9968bfe..0d6e31a 100644
--- a/example/timer/odp_timer_test.c
+++ b/example/timer/odp_timer_test.c
@@ -244,12 +244,12 @@ int main(int argc, char *argv[])
        test_args_t args;
        int num_workers;
        odp_buffer_pool_t pool;
-       void *pool_base;
        odp_queue_t queue;
        int first_core;
        uint64_t cycles, ns;
        odp_queue_param_t param;
        odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
        printf("\nODP timer example starts\n");
 
@@ -313,12 +313,13 @@ int main(int argc, char *argv[])
         */
        shm = odp_shm_reserve("msg_pool",
                              MSG_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
-       pool_base = odp_shm_addr(shm);
 
-       pool = odp_buffer_pool_create("msg_pool", pool_base, MSG_POOL_SIZE,
-                                     0,
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_TIMEOUT);
+       params.buf_size  = 0;
+       params.buf_align = 0;
+       params.num_bufs  = MSG_POOL_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_TIMEOUT;
+
+       pool = odp_buffer_pool_create("msg_pool", shm, &params);
 
        if (pool == ODP_BUFFER_POOL_INVALID) {
                EXAMPLE_ERR("Pool create failed.\n");
diff --git a/platform/linux-generic/include/api/odp_buffer_pool.h 
b/platform/linux-generic/include/api/odp_buffer_pool.h
index b287f79..4ccdfe6 100644
--- a/platform/linux-generic/include/api/odp_buffer_pool.h
+++ b/platform/linux-generic/include/api/odp_buffer_pool.h
@@ -33,21 +33,48 @@ extern "C" {
 #define ODP_BUFFER_POOL_NAME_LEN  32
 
 /**
+ * Buffer pool parameters
+ * Used to communicate buffer pool creation options.
+ */
+typedef struct odp_buffer_pool_param_t {
+       uint32_t buf_size;  /**< Buffer size in bytes.  The maximum
+                              number of bytes application will
+                              store in each buffer. For packets, this
+                              is the maximum packet data length, and
+                              configured headroom and tailroom will be
+                              added to this number */
+       uint32_t buf_align; /**< Minimum buffer alignment in bytes.
+                              Valid values are powers of two.  Use 0
+                              for default alignment.  Default will
+                              always be a multiple of 8. */
+       uint32_t num_bufs;  /**< Number of buffers in the pool */
+       int      buf_type;  /**< Buffer type */
+} odp_buffer_pool_param_t;
+
+/**
  * Create a buffer pool
+ * This routine is used to create a buffer pool. It take three
+ * arguments: the optional name of the pool to be created, an optional shared
+ * memory handle, and a parameter struct that describes the pool to be
+ * created. If a name is not specified the result is an anonymous pool that
+ * cannot be referenced by odp_buffer_pool_lookup().
  *
- * @param name      Name of the pool (max ODP_BUFFER_POOL_NAME_LEN - 1 chars)
- * @param base_addr Pool base address
- * @param size      Pool size in bytes
- * @param buf_size  Buffer size in bytes
- * @param buf_align Minimum buffer alignment
- * @param buf_type  Buffer type
+ * @param name     Name of the pool, max ODP_BUFFER_POOL_NAME_LEN-1 chars.
+ *                 May be specified as NULL for anonymous pools.
  *
- * @return Buffer pool handle
+ * @param shm      The shared memory object in which to create the pool.
+ *                 Use ODP_SHM_NULL to reserve default memory type
+ *                 for the buffer type.
+ *
+ * @param params   Buffer pool parameters.
+ *
+ * @return Handle of the created buffer pool
+ * @retval ODP_BUFFER_POOL_INVALID  Buffer pool could not be created
  */
+
 odp_buffer_pool_t odp_buffer_pool_create(const char *name,
-                                        void *base_addr, uint64_t size,
-                                        size_t buf_size, size_t buf_align,
-                                        int buf_type);
+                                        odp_shm_t shm,
+                                        odp_buffer_pool_param_t *params);
 
 
 /**
diff --git a/platform/linux-generic/include/api/odp_platform_types.h 
b/platform/linux-generic/include/api/odp_platform_types.h
index 68d0c55..2181eb6 100644
--- a/platform/linux-generic/include/api/odp_platform_types.h
+++ b/platform/linux-generic/include/api/odp_platform_types.h
@@ -27,7 +27,7 @@
 typedef uint32_t odp_buffer_pool_t;
 
 /** Invalid buffer pool */
-#define ODP_BUFFER_POOL_INVALID (0)
+#define ODP_BUFFER_POOL_INVALID (0xffffffff)
 
 /** ODP buffer */
 typedef uint32_t odp_buffer_t;
@@ -74,6 +74,7 @@ typedef uint32_t odp_shm_t;
 
 /** Invalid shared memory block */
 #define ODP_SHM_INVALID 0
+#define ODP_SHM_NULL ODP_SHM_INVALID /**< Synonym for buffer pool use */
 
 /**
  * @}
diff --git a/platform/linux-generic/include/odp_buffer_inlines.h 
b/platform/linux-generic/include/odp_buffer_inlines.h
new file mode 100644
index 0000000..6227482
--- /dev/null
+++ b/platform/linux-generic/include/odp_buffer_inlines.h
@@ -0,0 +1,150 @@
+/* Copyright (c) 2014, Linaro Limited
+ * All rights reserved.
+ *
+ * SPDX-License-Identifier:     BSD-3-Clause
+ */
+
+/**
+ * @file
+ *
+ * Inline functions for ODP buffer mgmt routines - implementation internal
+ */
+
+#ifndef ODP_BUFFER_INLINES_H_
+#define ODP_BUFFER_INLINES_H_
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+static inline odp_buffer_t odp_buffer_encode_handle(odp_buffer_hdr_t *hdr)
+{
+       odp_buffer_bits_t handle;
+       uint32_t pool_id = pool_handle_to_index(hdr->pool_hdl);
+       struct pool_entry_s *pool = get_pool_entry(pool_id);
+
+       handle.pool_id = pool_id;
+       handle.index = ((uint8_t *)hdr - pool->pool_mdata_addr) /
+               ODP_CACHE_LINE_SIZE;
+       handle.seg = 0;
+
+       return handle.u32;
+}
+
+static inline odp_buffer_t odp_hdr_to_buf(odp_buffer_hdr_t *hdr)
+{
+       return hdr->handle.handle;
+}
+
+static inline odp_buffer_hdr_t *odp_buf_to_hdr(odp_buffer_t buf)
+{
+       odp_buffer_bits_t handle;
+       uint32_t pool_id;
+       uint32_t index;
+       struct pool_entry_s *pool;
+
+       handle.u32 = buf;
+       pool_id    = handle.pool_id;
+       index      = handle.index;
+
+#ifdef POOL_ERROR_CHECK
+       if (odp_unlikely(pool_id > ODP_CONFIG_BUFFER_POOLS)) {
+               ODP_ERR("odp_buf_to_hdr: Bad pool id\n");
+               return NULL;
+       }
+#endif
+
+       pool = get_pool_entry(pool_id);
+
+#ifdef POOL_ERROR_CHECK
+       if (odp_unlikely(index > pool->params.num_bufs - 1)) {
+               ODP_ERR("odp_buf_to_hdr: Bad buffer index\n");
+               return NULL;
+       }
+#endif
+
+       return (odp_buffer_hdr_t *)(void *)
+               (pool->pool_mdata_addr + (index * ODP_CACHE_LINE_SIZE));
+}
+
+static inline uint32_t odp_buffer_refcount(odp_buffer_hdr_t *buf)
+{
+       return odp_atomic_load_u32(&buf->ref_count);
+}
+
+static inline uint32_t odp_buffer_incr_refcount(odp_buffer_hdr_t *buf,
+                                               uint32_t val)
+{
+       return odp_atomic_fetch_add_u32(&buf->ref_count, val) + val;
+}
+
+static inline uint32_t odp_buffer_decr_refcount(odp_buffer_hdr_t *buf,
+                                               uint32_t val)
+{
+       uint32_t tmp;
+
+       tmp = odp_atomic_fetch_sub_u32(&buf->ref_count, val);
+
+       if (tmp < val) {
+               odp_atomic_fetch_add_u32(&buf->ref_count, val - tmp);
+               return 0;
+       } else {
+               return tmp - val;
+       }
+}
+
+static inline odp_buffer_hdr_t *validate_buf(odp_buffer_t buf)
+{
+       odp_buffer_bits_t handle;
+       odp_buffer_hdr_t *buf_hdr;
+       handle.u32 = buf;
+
+       /* For buffer handles, segment index must be 0 and pool id in range */
+       if (handle.seg != 0 || handle.pool_id >= ODP_CONFIG_BUFFER_POOLS)
+               return NULL;
+
+       pool_entry_t *pool = odp_pool_to_entry(handle.pool_id);
+
+       /* If pool not created, handle is invalid */
+       if (pool->s.pool_shm == ODP_SHM_INVALID)
+               return NULL;
+
+       uint32_t buf_stride = pool->s.buf_stride / ODP_CACHE_LINE_SIZE;
+
+       /* A valid buffer index must be on stride, and must be in range */
+       if ((handle.index % buf_stride != 0) ||
+           ((uint32_t)(handle.index / buf_stride) >= pool->s.params.num_bufs))
+               return NULL;
+
+       buf_hdr = (odp_buffer_hdr_t *)(void *)
+               (pool->s.pool_mdata_addr +
+                (handle.index * ODP_CACHE_LINE_SIZE));
+
+       /* Handle is valid, so buffer is valid if it is allocated */
+       return buf_hdr->allocator == ODP_FREEBUF ? NULL : buf_hdr;
+}
+
+int odp_buffer_snprint(char *str, uint32_t n, odp_buffer_t buf);
+
+static inline void *buffer_map(odp_buffer_hdr_t *buf,
+                              uint32_t offset,
+                              uint32_t *seglen,
+                              uint32_t limit)
+{
+       int seg_index  = offset / buf->segsize;
+       int seg_offset = offset % buf->segsize;
+
+       if (seglen != NULL) {
+               uint32_t buf_left = limit - offset;
+               *seglen = buf_left < buf->segsize ?
+                       buf_left : buf->segsize - seg_offset;
+       }
+
+       return (void *)(seg_offset + (uint8_t *)buf->addr[seg_index]);
+}
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
diff --git a/platform/linux-generic/include/odp_buffer_internal.h 
b/platform/linux-generic/include/odp_buffer_internal.h
index 0027bfc..859633e 100644
--- a/platform/linux-generic/include/odp_buffer_internal.h
+++ b/platform/linux-generic/include/odp_buffer_internal.h
@@ -24,99 +24,133 @@ extern "C" {
 #include <odp_buffer.h>
 #include <odp_debug.h>
 #include <odp_align.h>
-
-/* TODO: move these to correct files */
-
-typedef uint64_t odp_phys_addr_t;
+#include <odp_align_internal.h>
+#include <odp_config.h>
+#include <odp_byteorder.h>
+#include <odp_thread.h>
+
+
+#define ODP_BITSIZE(x) \
+       ((x) <=     2 ?  1 : \
+       ((x) <=     4 ?  2 : \
+       ((x) <=     8 ?  3 : \
+       ((x) <=    16 ?  4 : \
+       ((x) <=    32 ?  5 : \
+       ((x) <=    64 ?  6 : \
+       ((x) <=   128 ?  7 : \
+       ((x) <=   256 ?  8 : \
+       ((x) <=   512 ?  9 : \
+       ((x) <=  1024 ? 10 : \
+       ((x) <=  2048 ? 11 : \
+       ((x) <=  4096 ? 12 : \
+       ((x) <=  8196 ? 13 : \
+       ((x) <= 16384 ? 14 : \
+       ((x) <= 32768 ? 15 : \
+       ((x) <= 65536 ? 16 : \
+        (0/0)))))))))))))))))
+
+ODP_STATIC_ASSERT(ODP_CONFIG_PACKET_BUF_LEN_MIN >= 256,
+                 "ODP Segment size must be a minimum of 256 bytes");
+
+ODP_STATIC_ASSERT((ODP_CONFIG_PACKET_BUF_LEN_MIN % ODP_CACHE_LINE_SIZE) == 0,
+                 "ODP Segment size must be a multiple of cache line size");
+
+ODP_STATIC_ASSERT((ODP_CONFIG_PACKET_BUF_LEN_MAX %
+                  ODP_CONFIG_PACKET_BUF_LEN_MIN) == 0,
+                 "Packet max size must be a multiple of segment size");
+
+#define ODP_BUFFER_MAX_SEG \
+       (ODP_CONFIG_PACKET_BUF_LEN_MAX / ODP_CONFIG_PACKET_BUF_LEN_MIN)
+
+/* We can optimize storage of small raw buffers within metadata area */
+#define ODP_MAX_INLINE_BUF     ((sizeof(void *)) * (ODP_BUFFER_MAX_SEG - 1))
+
+#define ODP_BUFFER_POOL_BITS   ODP_BITSIZE(ODP_CONFIG_BUFFER_POOLS)
+#define ODP_BUFFER_SEG_BITS    ODP_BITSIZE(ODP_BUFFER_MAX_SEG)
+#define ODP_BUFFER_INDEX_BITS  (32 - ODP_BUFFER_POOL_BITS - 
ODP_BUFFER_SEG_BITS)
+#define ODP_BUFFER_PREFIX_BITS (ODP_BUFFER_POOL_BITS + ODP_BUFFER_INDEX_BITS)
+#define ODP_BUFFER_MAX_POOLS   (1 << ODP_BUFFER_POOL_BITS)
+#define ODP_BUFFER_MAX_BUFFERS (1 << ODP_BUFFER_INDEX_BITS)
 
 #define ODP_BUFFER_MAX_INDEX     (ODP_BUFFER_MAX_BUFFERS - 2)
 #define ODP_BUFFER_INVALID_INDEX (ODP_BUFFER_MAX_BUFFERS - 1)
 
-#define ODP_BUFS_PER_CHUNK       16
-#define ODP_BUFS_PER_SCATTER      4
-
-#define ODP_BUFFER_TYPE_CHUNK    0xffff
-
-
-#define ODP_BUFFER_POOL_BITS   4
-#define ODP_BUFFER_INDEX_BITS  (32 - ODP_BUFFER_POOL_BITS)
-#define ODP_BUFFER_MAX_POOLS   (1 << ODP_BUFFER_POOL_BITS)
-#define ODP_BUFFER_MAX_BUFFERS (1 << ODP_BUFFER_INDEX_BITS)
-
 typedef union odp_buffer_bits_t {
        uint32_t     u32;
        odp_buffer_t handle;
 
        struct {
+#if ODP_BYTE_ORDER == ODP_BIG_ENDIAN
                uint32_t pool_id:ODP_BUFFER_POOL_BITS;
                uint32_t index:ODP_BUFFER_INDEX_BITS;
+               uint32_t seg:ODP_BUFFER_SEG_BITS;
+#else
+               uint32_t seg:ODP_BUFFER_SEG_BITS;
+               uint32_t index:ODP_BUFFER_INDEX_BITS;
+               uint32_t pool_id:ODP_BUFFER_POOL_BITS;
+#endif
        };
-} odp_buffer_bits_t;
 
+       struct {
+#if ODP_BYTE_ORDER == ODP_BIG_ENDIAN
+               uint32_t prefix:ODP_BUFFER_PREFIX_BITS;
+               uint32_t pfxseg:ODP_BUFFER_SEG_BITS;
+#else
+               uint32_t pfxseg:ODP_BUFFER_SEG_BITS;
+               uint32_t prefix:ODP_BUFFER_PREFIX_BITS;
+#endif
+       };
+} odp_buffer_bits_t;
 
 /* forward declaration */
 struct odp_buffer_hdr_t;
 
-
-/*
- * Scatter/gather list of buffers
- */
-typedef struct odp_buffer_scatter_t {
-       /* buffer pointers */
-       struct odp_buffer_hdr_t *buf[ODP_BUFS_PER_SCATTER];
-       int                      num_bufs;   /* num buffers */
-       int                      pos;        /* position on the list */
-       size_t                   total_len;  /* Total length */
-} odp_buffer_scatter_t;
-
-
-/*
- * Chunk of buffers (in single pool)
- */
-typedef struct odp_buffer_chunk_t {
-       uint32_t num_bufs;                      /* num buffers */
-       uint32_t buf_index[ODP_BUFS_PER_CHUNK]; /* buffers */
-} odp_buffer_chunk_t;
-
-
 /* Common buffer header */
 typedef struct odp_buffer_hdr_t {
        struct odp_buffer_hdr_t *next;       /* next buf in a list */
+       int                      allocator;  /* allocating thread id */
        odp_buffer_bits_t        handle;     /* handle */
-       odp_phys_addr_t          phys_addr;  /* physical data start address */
-       void                    *addr;       /* virtual data start address */
-       uint32_t                 index;      /* buf index in the pool */
+       union {
+               uint32_t all;
+               struct {
+                       uint32_t zeroized:1; /* Zeroize buf data on free */
+                       uint32_t hdrdata:1;  /* Data is in buffer hdr */
+               };
+       } flags;
+       int                      type;       /* buffer type */
        size_t                   size;       /* max data size */
-       size_t                   cur_offset; /* current offset */
        odp_atomic_u32_t         ref_count;  /* reference count */
-       odp_buffer_scatter_t     scatter;    /* Scatter/gather list */
-       int                      type;       /* type of next header */
        odp_buffer_pool_t        pool_hdl;   /* buffer pool handle */
-
+       union {
+               uint64_t         buf_u64;    /* user u64 */
+               void            *buf_ctx;    /* user context */
+               void            *udata_addr; /* user metadata addr */
+       };
+       size_t                   udata_size; /* size of user metadata */
+       uint32_t                 segcount;   /* segment count */
+       uint32_t                 segsize;    /* segment size */
+       void                    *addr[ODP_BUFFER_MAX_SEG]; /* block addrs */
 } odp_buffer_hdr_t;
 
-/* Ensure next header starts from 8 byte align */
-ODP_STATIC_ASSERT((sizeof(odp_buffer_hdr_t) % 8) == 0, 
"ODP_BUFFER_HDR_T__SIZE_ERROR");
+typedef struct odp_buffer_hdr_stride {
+       uint8_t pad[ODP_CACHE_LINE_SIZE_ROUNDUP(sizeof(odp_buffer_hdr_t))];
+} odp_buffer_hdr_stride;
 
+typedef struct odp_buf_blk_t {
+       struct odp_buf_blk_t *next;
+       struct odp_buf_blk_t *prev;
+} odp_buf_blk_t;
 
 /* Raw buffer header */
 typedef struct {
        odp_buffer_hdr_t buf_hdr;    /* common buffer header */
-       uint8_t          buf_data[]; /* start of buffer data area */
 } odp_raw_buffer_hdr_t;
 
+/* Free buffer marker */
+#define ODP_FREEBUF -1
 
-/* Chunk header */
-typedef struct odp_buffer_chunk_hdr_t {
-       odp_buffer_hdr_t   buf_hdr;
-       odp_buffer_chunk_t chunk;
-} odp_buffer_chunk_hdr_t;
-
-
-int odp_buffer_snprint(char *str, size_t n, odp_buffer_t buf);
-
-void odp_buffer_copy_scatter(odp_buffer_t buf_dst, odp_buffer_t buf_src);
-
+/* Forward declarations */
+odp_buffer_t buffer_alloc(odp_buffer_pool_t pool, size_t size);
 
 #ifdef __cplusplus
 }
diff --git a/platform/linux-generic/include/odp_buffer_pool_internal.h 
b/platform/linux-generic/include/odp_buffer_pool_internal.h
index 07602fe..2e48ac3 100644
--- a/platform/linux-generic/include/odp_buffer_pool_internal.h
+++ b/platform/linux-generic/include/odp_buffer_pool_internal.h
@@ -19,13 +19,44 @@ extern "C" {
 #endif
 
 #include <odp_std_types.h>
-#include <odp_buffer_pool.h>
-#include <odp_buffer_internal.h>
 #include <odp_align.h>
 #include <odp_align_internal.h>
+#include <odp_buffer_pool.h>
+#include <odp_buffer_internal.h>
 #include <odp_hints.h>
 #include <odp_config.h>
 #include <odp_debug.h>
+#include <odp_shared_memory.h>
+#include <odp_atomic.h>
+#include <odp_atomic_internal.h>
+#include <string.h>
+
+/**
+ * Buffer initialization routine prototype
+ *
+ * @note Routines of this type MAY be passed as part of the
+ * _odp_buffer_pool_init_t structure to be called whenever a
+ * buffer is allocated to initialize the user metadata
+ * associated with that buffer.
+ */
+typedef void (_odp_buf_init_t)(odp_buffer_t buf, void *buf_init_arg);
+
+/**
+ * Buffer pool initialization parameters
+ * Used to communicate buffer pool initialization options. Internal for now.
+ */
+typedef struct _odp_buffer_pool_init_t {
+       size_t udata_size;         /**< Size of user metadata for each buffer */
+       _odp_buf_init_t *buf_init; /**< Buffer initialization routine to use */
+       void *buf_init_arg;        /**< Argument to be passed to buf_init() */
+} _odp_buffer_pool_init_t;         /**< Type of buffer initialization struct */
+
+/* Local cache for buffer alloc/free acceleration */
+typedef struct local_cache_t {
+       odp_buffer_hdr_t *buf_freelist;  /* The local cache */
+       uint64_t bufallocs;              /* Local buffer alloc count */
+       uint64_t buffrees;               /* Local buffer free count */
+} local_cache_t;
 
 /* Use ticketlock instead of spinlock */
 #define POOL_USE_TICKETLOCK
@@ -36,11 +67,16 @@ extern "C" {
 
 #ifdef POOL_USE_TICKETLOCK
 #include <odp_ticketlock.h>
+#define POOL_LOCK(a)      odp_ticketlock_lock(a)
+#define POOL_UNLOCK(a)    odp_ticketlock_unlock(a)
+#define POOL_LOCK_INIT(a) odp_ticketlock_init(a)
 #else
 #include <odp_spinlock.h>
+#define POOL_LOCK(a)      odp_spinlock_lock(a)
+#define POOL_UNLOCK(a)    odp_spinlock_unlock(a)
+#define POOL_LOCK_INIT(a) odp_spinlock_init(a)
 #endif
 
-
 struct pool_entry_s {
 #ifdef POOL_USE_TICKETLOCK
        odp_ticketlock_t        lock ODP_ALIGNED_CACHE;
@@ -48,74 +84,294 @@ struct pool_entry_s {
        odp_spinlock_t          lock ODP_ALIGNED_CACHE;
 #endif
 
-       odp_buffer_chunk_hdr_t *head;
-       uint64_t                free_bufs;
        char                    name[ODP_BUFFER_POOL_NAME_LEN];
-
-       odp_buffer_pool_t       pool_hdl ODP_ALIGNED_CACHE;
-       uintptr_t               buf_base;
-       size_t                  buf_size;
-       size_t                  buf_offset;
-       uint64_t                num_bufs;
-       void                   *pool_base_addr;
-       uint64_t                pool_size;
-       size_t                  user_size;
-       size_t                  user_align;
-       int                     buf_type;
-       size_t                  hdr_size;
+       odp_buffer_pool_param_t params;
+       _odp_buffer_pool_init_t init_params;
+       odp_buffer_pool_t       pool_hdl;
+       uint32_t                pool_id;
+       odp_shm_t               pool_shm;
+       union {
+               uint32_t all;
+               struct {
+                       uint32_t has_name:1;
+                       uint32_t user_supplied_shm:1;
+                       uint32_t unsegmented:1;
+                       uint32_t zeroized:1;
+                       uint32_t predefined:1;
+               };
+       } flags;
+       uint32_t                quiesced;
+       uint32_t                low_wm_assert;
+       uint8_t                *pool_base_addr;
+       uint8_t                *pool_mdata_addr;
+       size_t                  pool_size;
+       uint32_t                buf_align;
+       uint32_t                buf_stride;
+       _odp_atomic_ptr_t       buf_freelist;
+       _odp_atomic_ptr_t       blk_freelist;
+       odp_atomic_u32_t        bufcount;
+       odp_atomic_u32_t        blkcount;
+       odp_atomic_u64_t        bufallocs;
+       odp_atomic_u64_t        buffrees;
+       odp_atomic_u64_t        blkallocs;
+       odp_atomic_u64_t        blkfrees;
+       odp_atomic_u64_t        bufempty;
+       odp_atomic_u64_t        blkempty;
+       odp_atomic_u64_t        high_wm_count;
+       odp_atomic_u64_t        low_wm_count;
+       uint32_t                seg_size;
+       uint32_t                high_wm;
+       uint32_t                low_wm;
+       uint32_t                headroom;
+       uint32_t                tailroom;
 };
 
 typedef union pool_entry_u {
        struct pool_entry_s s;
+
        uint8_t pad[ODP_CACHE_LINE_SIZE_ROUNDUP(sizeof(struct pool_entry_s))];
 } pool_entry_t;
 
 extern void *pool_entry_ptr[];
 
+#if defined(ODP_CONFIG_SECURE_POOLS) && (ODP_CONFIG_SECURE_POOLS == 1)
+#define buffer_is_secure(buf) (buf->flags.zeroized)
+#define pool_is_secure(pool) (pool->flags.zeroized)
+#else
+#define buffer_is_secure(buf) 0
+#define pool_is_secure(pool) 0
+#endif
 
-static inline void *get_pool_entry(uint32_t pool_id)
+#define TAG_ALIGN ((size_t)16)
+
+#define odp_cs(ptr, old, new) \
+       _odp_atomic_ptr_cmp_xchg_strong(&ptr, (void **)&old, (void *)new, \
+                                       _ODP_MEMMODEL_SC, \
+                                       _ODP_MEMMODEL_SC)
+
+/* Helper functions for pointer tagging to avoid ABA race conditions */
+#define odp_tag(ptr) \
+       (((size_t)ptr) & (TAG_ALIGN - 1))
+
+#define odp_detag(ptr) \
+       ((void *)(((size_t)ptr) & -TAG_ALIGN))
+
+#define odp_retag(ptr, tag) \
+       ((void *)(((size_t)ptr) | odp_tag(tag)))
+
+
+static inline void *get_blk(struct pool_entry_s *pool)
 {
-       return pool_entry_ptr[pool_id];
+       void *oldhead, *myhead, *newhead;
+
+       oldhead = _odp_atomic_ptr_load(&pool->blk_freelist, _ODP_MEMMODEL_ACQ);
+
+       do {
+               size_t tag = odp_tag(oldhead);
+               myhead = odp_detag(oldhead);
+               if (odp_unlikely(myhead == NULL))
+                       break;
+               newhead = odp_retag(((odp_buf_blk_t *)myhead)->next, tag + 1);
+       } while (odp_cs(pool->blk_freelist, oldhead, newhead) == 0);
+
+       if (odp_unlikely(myhead == NULL))
+               odp_atomic_inc_u64(&pool->blkempty);
+       else
+               odp_atomic_dec_u32(&pool->blkcount);
+
+       return (void *)myhead;
 }
 
-static inline uint32_t pool_handle_to_index(odp_buffer_pool_t pool_hdl)
+static inline void ret_blk(struct pool_entry_s *pool, void *block)
 {
-       return pool_hdl - 1;
+       void *oldhead, *myhead, *myblock;
+
+       oldhead = _odp_atomic_ptr_load(&pool->blk_freelist, _ODP_MEMMODEL_ACQ);
+
+       do {
+               size_t tag = odp_tag(oldhead);
+               myhead = odp_detag(oldhead);
+               ((odp_buf_blk_t *)block)->next = myhead;
+               myblock = odp_retag(block, tag + 1);
+       } while (odp_cs(pool->blk_freelist, oldhead, myblock) == 0);
+
+       odp_atomic_inc_u32(&pool->blkcount);
+       odp_atomic_inc_u64(&pool->blkfrees);
 }
 
-static inline odp_buffer_hdr_t *odp_buf_to_hdr(odp_buffer_t buf)
+static inline odp_buffer_hdr_t *get_buf(struct pool_entry_s *pool)
 {
-       odp_buffer_bits_t handle;
-       uint32_t pool_id;
-       uint32_t index;
-       struct pool_entry_s *pool;
-       odp_buffer_hdr_t *hdr;
-
-       handle.u32 = buf;
-       pool_id    = handle.pool_id;
-       index      = handle.index;
-
-#ifdef POOL_ERROR_CHECK
-       if (odp_unlikely(pool_id > ODP_CONFIG_BUFFER_POOLS)) {
-               ODP_ERR("odp_buf_to_hdr: Bad pool id\n");
-               return NULL;
+       odp_buffer_hdr_t *oldhead, *myhead, *newhead;
+
+       oldhead = _odp_atomic_ptr_load(&pool->buf_freelist, _ODP_MEMMODEL_ACQ);
+
+       do {
+               size_t tag = odp_tag(oldhead);
+               myhead = odp_detag(oldhead);
+               if (odp_unlikely(myhead == NULL))
+                       break;
+               newhead = odp_retag(myhead->next, tag + 1);
+       } while (odp_cs(pool->buf_freelist, oldhead, newhead) == 0);
+
+       if (odp_unlikely(myhead == NULL)) {
+               odp_atomic_inc_u64(&pool->bufempty);
+       } else {
+               uint64_t bufcount =
+                       odp_atomic_fetch_sub_u32(&pool->bufcount, 1) - 1;
+
+               /* Check for low watermark condition */
+               if (bufcount == pool->low_wm && !pool->low_wm_assert) {
+                       pool->low_wm_assert = 1;
+                       odp_atomic_inc_u64(&pool->low_wm_count);
+               }
+
+               odp_atomic_inc_u64(&pool->bufallocs);
+               myhead->next = myhead;  /* Mark buffer allocated */
+               myhead->allocator = odp_thread_id();
        }
-#endif
 
-       pool = get_pool_entry(pool_id);
+       return (void *)myhead;
+}
+
+static inline void ret_buf(struct pool_entry_s *pool, odp_buffer_hdr_t *buf)
+{
+       odp_buffer_hdr_t *oldhead, *myhead, *mybuf;
+
+       buf->allocator = ODP_FREEBUF;  /* Mark buffer free */
 
-#ifdef POOL_ERROR_CHECK
-       if (odp_unlikely(index > pool->num_bufs - 1)) {
-               ODP_ERR("odp_buf_to_hdr: Bad buffer index\n");
-               return NULL;
+       if (!buf->flags.hdrdata && buf->type != ODP_BUFFER_TYPE_RAW) {
+               while (buf->segcount > 0) {
+                       if (buffer_is_secure(buf) || pool_is_secure(pool))
+                               memset(buf->addr[buf->segcount - 1],
+                                      0, buf->segsize);
+                       ret_blk(pool, buf->addr[--buf->segcount]);
+               }
+               buf->size = 0;
        }
-#endif
 
-       hdr = (odp_buffer_hdr_t *)(pool->buf_base + index * pool->buf_size);
+       oldhead = _odp_atomic_ptr_load(&pool->buf_freelist, _ODP_MEMMODEL_ACQ);
+
+       do {
+               size_t tag = odp_tag(oldhead);
+               myhead = odp_detag(oldhead);
+               buf->next = myhead;
+               mybuf = odp_retag(buf, tag + 1);
+       } while (odp_cs(pool->buf_freelist, oldhead, mybuf) == 0);
+
+       uint64_t bufcount = odp_atomic_fetch_add_u32(&pool->bufcount, 1) + 1;
+
+       /* Check if low watermark condition should be deasserted */
+       if (bufcount == pool->high_wm && pool->low_wm_assert) {
+               pool->low_wm_assert = 0;
+               odp_atomic_inc_u64(&pool->high_wm_count);
+       }
 
-       return hdr;
+       odp_atomic_inc_u64(&pool->buffrees);
 }
 
+static inline void *get_local_buf(local_cache_t *buf_cache,
+                                 struct pool_entry_s *pool,
+                                 size_t totsize)
+{
+       odp_buffer_hdr_t *buf = buf_cache->buf_freelist;
+
+       if (odp_likely(buf != NULL)) {
+               buf_cache->buf_freelist = buf->next;
+
+               if (odp_unlikely(buf->size < totsize)) {
+                       intmax_t needed = totsize - buf->size;
+
+                       do {
+                               void *blk = get_blk(pool);
+                               if (odp_unlikely(blk == NULL)) {
+                                       ret_buf(pool, buf);
+                                       buf_cache->buffrees--;
+                                       return NULL;
+                               }
+                               buf->addr[buf->segcount++] = blk;
+                               needed -= pool->seg_size;
+                       } while (needed > 0);
+
+                       buf->size = buf->segcount * pool->seg_size;
+               }
+
+               buf_cache->bufallocs++;
+               buf->allocator = odp_thread_id();  /* Mark buffer allocated */
+       }
+
+       return buf;
+}
+
+static inline void ret_local_buf(local_cache_t *buf_cache,
+                               odp_buffer_hdr_t *buf)
+{
+       buf->allocator = ODP_FREEBUF;
+       buf->next = buf_cache->buf_freelist;
+       buf_cache->buf_freelist = buf;
+
+       buf_cache->buffrees++;
+}
+
+static inline void flush_cache(local_cache_t *buf_cache,
+                              struct pool_entry_s *pool)
+{
+       odp_buffer_hdr_t *buf = buf_cache->buf_freelist;
+       uint32_t flush_count = 0;
+
+       while (buf != NULL) {
+               odp_buffer_hdr_t *next = buf->next;
+               ret_buf(pool, buf);
+               buf = next;
+               flush_count++;
+       }
+
+       odp_atomic_add_u64(&pool->bufallocs, buf_cache->bufallocs);
+       odp_atomic_add_u64(&pool->buffrees, buf_cache->buffrees - flush_count);
+
+       buf_cache->buf_freelist = NULL;
+       buf_cache->bufallocs = 0;
+       buf_cache->buffrees = 0;
+}
+
+static inline odp_buffer_pool_t pool_index_to_handle(uint32_t pool_id)
+{
+       return pool_id;
+}
+
+static inline uint32_t pool_handle_to_index(odp_buffer_pool_t pool_hdl)
+{
+       return pool_hdl;
+}
+
+static inline void *get_pool_entry(uint32_t pool_id)
+{
+       return pool_entry_ptr[pool_id];
+}
+
+static inline pool_entry_t *odp_pool_to_entry(odp_buffer_pool_t pool)
+{
+       return (pool_entry_t *)get_pool_entry(pool_handle_to_index(pool));
+}
+
+static inline pool_entry_t *odp_buf_to_pool(odp_buffer_hdr_t *buf)
+{
+       return odp_pool_to_entry(buf->pool_hdl);
+}
+
+static inline uint32_t odp_buffer_pool_segment_size(odp_buffer_pool_t pool)
+{
+       return odp_pool_to_entry(pool)->s.seg_size;
+}
+
+static inline uint32_t odp_buffer_pool_headroom(odp_buffer_pool_t pool)
+{
+       return odp_pool_to_entry(pool)->s.headroom;
+}
+
+static inline uint32_t odp_buffer_pool_tailroom(odp_buffer_pool_t pool)
+{
+       return odp_pool_to_entry(pool)->s.tailroom;
+}
 
 #ifdef __cplusplus
 }
diff --git a/platform/linux-generic/include/odp_internal.h 
b/platform/linux-generic/include/odp_internal.h
index 04c1030..ee2ab1a 100644
--- a/platform/linux-generic/include/odp_internal.h
+++ b/platform/linux-generic/include/odp_internal.h
@@ -44,6 +44,8 @@ int odp_schedule_init_local(void);
 int odp_timer_init_global(void);
 int odp_timer_disarm_all(void);
 
+void _odp_flush_caches(void);
+
 #ifdef __cplusplus
 }
 #endif
diff --git a/platform/linux-generic/include/odp_packet_internal.h 
b/platform/linux-generic/include/odp_packet_internal.h
index 49c59b2..f34a83d 100644
--- a/platform/linux-generic/include/odp_packet_internal.h
+++ b/platform/linux-generic/include/odp_packet_internal.h
@@ -22,6 +22,7 @@ extern "C" {
 #include <odp_debug.h>
 #include <odp_buffer_internal.h>
 #include <odp_buffer_pool_internal.h>
+#include <odp_buffer_inlines.h>
 #include <odp_packet.h>
 #include <odp_packet_io.h>
 
@@ -92,7 +93,8 @@ typedef union {
        };
 } output_flags_t;
 
-ODP_STATIC_ASSERT(sizeof(output_flags_t) == sizeof(uint32_t), 
"OUTPUT_FLAGS_SIZE_ERROR");
+ODP_STATIC_ASSERT(sizeof(output_flags_t) == sizeof(uint32_t),
+                 "OUTPUT_FLAGS_SIZE_ERROR");
 
 /**
  * Internal Packet header
@@ -105,25 +107,23 @@ typedef struct {
        error_flags_t  error_flags;
        output_flags_t output_flags;
 
-       uint32_t frame_offset; /**< offset to start of frame, even on error */
        uint32_t l2_offset; /**< offset to L2 hdr, e.g. Eth */
        uint32_t l3_offset; /**< offset to L3 hdr, e.g. IPv4, IPv6 */
        uint32_t l4_offset; /**< offset to L4 hdr (TCP, UDP, SCTP, also ICMP) */
 
        uint32_t frame_len;
+       uint32_t headroom;
+       uint32_t tailroom;
 
        uint64_t user_ctx;        /* user context */
 
        odp_pktio_t input;
-
-       uint32_t pad;
-       uint8_t  buf_data[]; /* start of buffer data area */
 } odp_packet_hdr_t;
 
-ODP_STATIC_ASSERT(sizeof(odp_packet_hdr_t) == ODP_OFFSETOF(odp_packet_hdr_t, 
buf_data),
-          "ODP_PACKET_HDR_T__SIZE_ERR");
-ODP_STATIC_ASSERT(sizeof(odp_packet_hdr_t) % sizeof(uint64_t) == 0,
-          "ODP_PACKET_HDR_T__SIZE_ERR2");
+typedef struct odp_packet_hdr_stride {
+       uint8_t pad[ODP_CACHE_LINE_SIZE_ROUNDUP(sizeof(odp_packet_hdr_t))];
+} odp_packet_hdr_stride;
+
 
 /**
  * Return the packet header
@@ -138,6 +138,38 @@ static inline odp_packet_hdr_t 
*odp_packet_hdr(odp_packet_t pkt)
  */
 void odp_packet_parse(odp_packet_t pkt, size_t len, size_t l2_offset);
 
+/**
+ * Initialize packet buffer
+ */
+static inline void packet_init(pool_entry_t *pool,
+                              odp_packet_hdr_t *pkt_hdr,
+                              size_t size)
+{
+       /*
+       * Reset parser metadata.  Note that we clear via memset to make
+       * this routine indepenent of any additional adds to packet metadata.
+       */
+       const size_t start_offset = ODP_FIELD_SIZEOF(odp_packet_hdr_t, buf_hdr);
+       uint8_t *start;
+       size_t len;
+
+       start = (uint8_t *)pkt_hdr + start_offset;
+       len = sizeof(odp_packet_hdr_t) - start_offset;
+       memset(start, 0, len);
+
+       /*
+       * Packet headroom is set from the pool's headroom
+       * Packet tailroom is rounded up to fill the last
+       * segment occupied by the allocated length.
+       */
+       pkt_hdr->frame_len = size;
+       pkt_hdr->headroom  = pool->s.headroom;
+       pkt_hdr->tailroom  =
+               (pool->s.seg_size * pkt_hdr->buf_hdr.segcount) -
+               (pool->s.headroom + size);
+}
+
+
 #ifdef __cplusplus
 }
 #endif
diff --git a/platform/linux-generic/include/odp_timer_internal.h 
b/platform/linux-generic/include/odp_timer_internal.h
index ad28f53..2ff36ce 100644
--- a/platform/linux-generic/include/odp_timer_internal.h
+++ b/platform/linux-generic/include/odp_timer_internal.h
@@ -51,14 +51,9 @@ typedef struct odp_timeout_hdr_t {
        uint8_t buf_data[];
 } odp_timeout_hdr_t;
 
-
-
-ODP_STATIC_ASSERT(sizeof(odp_timeout_hdr_t) ==
-          ODP_OFFSETOF(odp_timeout_hdr_t, buf_data),
-          "ODP_TIMEOUT_HDR_T__SIZE_ERR");
-
-ODP_STATIC_ASSERT(sizeof(odp_timeout_hdr_t) % sizeof(uint64_t) == 0,
-          "ODP_TIMEOUT_HDR_T__SIZE_ERR2");
+typedef struct odp_timeout_hdr_stride {
+       uint8_t pad[ODP_CACHE_LINE_SIZE_ROUNDUP(sizeof(odp_timeout_hdr_t))];
+} odp_timeout_hdr_stride;
 
 
 /**
diff --git a/platform/linux-generic/odp_buffer.c 
b/platform/linux-generic/odp_buffer.c
index df68aa9..dd37ab3 100644
--- a/platform/linux-generic/odp_buffer.c
+++ b/platform/linux-generic/odp_buffer.c
@@ -5,8 +5,9 @@
  */
 
 #include <odp_buffer.h>
-#include <odp_buffer_internal.h>
 #include <odp_buffer_pool_internal.h>
+#include <odp_buffer_internal.h>
+#include <odp_buffer_inlines.h>
 #include <odp_debug_internal.h>
 
 #include <string.h>
@@ -17,7 +18,7 @@ void *odp_buffer_addr(odp_buffer_t buf)
 {
        odp_buffer_hdr_t *hdr = odp_buf_to_hdr(buf);
 
-       return hdr->addr;
+       return hdr->addr[0];
 }
 
 
@@ -39,15 +40,11 @@ int odp_buffer_type(odp_buffer_t buf)
 
 int odp_buffer_is_valid(odp_buffer_t buf)
 {
-       odp_buffer_bits_t handle;
-
-       handle.u32 = buf;
-
-       return (handle.index != ODP_BUFFER_INVALID_INDEX);
+       return validate_buf(buf) != NULL;
 }
 
 
-int odp_buffer_snprint(char *str, size_t n, odp_buffer_t buf)
+int odp_buffer_snprint(char *str, uint32_t n, odp_buffer_t buf)
 {
        odp_buffer_hdr_t *hdr;
        int len = 0;
@@ -64,28 +61,14 @@ int odp_buffer_snprint(char *str, size_t n, odp_buffer_t 
buf)
        len += snprintf(&str[len], n-len,
                        "  pool         %i\n",        hdr->pool_hdl);
        len += snprintf(&str[len], n-len,
-                       "  index        %"PRIu32"\n", hdr->index);
-       len += snprintf(&str[len], n-len,
-                       "  phy_addr     %"PRIu64"\n", hdr->phys_addr);
-       len += snprintf(&str[len], n-len,
                        "  addr         %p\n",        hdr->addr);
        len += snprintf(&str[len], n-len,
                        "  size         %zu\n",       hdr->size);
        len += snprintf(&str[len], n-len,
-                       "  cur_offset   %zu\n",       hdr->cur_offset);
-       len += snprintf(&str[len], n-len,
                        "  ref_count    %i\n",
                        odp_atomic_load_u32(&hdr->ref_count));
        len += snprintf(&str[len], n-len,
                        "  type         %i\n",        hdr->type);
-       len += snprintf(&str[len], n-len,
-                       "  Scatter list\n");
-       len += snprintf(&str[len], n-len,
-                       "    num_bufs   %i\n",        hdr->scatter.num_bufs);
-       len += snprintf(&str[len], n-len,
-                       "    pos        %i\n",        hdr->scatter.pos);
-       len += snprintf(&str[len], n-len,
-                       "    total_len  %zu\n",       hdr->scatter.total_len);
 
        return len;
 }
@@ -102,9 +85,3 @@ void odp_buffer_print(odp_buffer_t buf)
 
        ODP_PRINT("\n%s\n", str);
 }
-
-void odp_buffer_copy_scatter(odp_buffer_t buf_dst, odp_buffer_t buf_src)
-{
-       (void)buf_dst;
-       (void)buf_src;
-}
diff --git a/platform/linux-generic/odp_buffer_pool.c 
b/platform/linux-generic/odp_buffer_pool.c
index d20999b..f19bee0 100644
--- a/platform/linux-generic/odp_buffer_pool.c
+++ b/platform/linux-generic/odp_buffer_pool.c
@@ -6,8 +6,9 @@
 
 #include <odp_std_types.h>
 #include <odp_buffer_pool.h>
-#include <odp_buffer_pool_internal.h>
 #include <odp_buffer_internal.h>
+#include <odp_buffer_pool_internal.h>
+#include <odp_buffer_inlines.h>
 #include <odp_packet_internal.h>
 #include <odp_timer_internal.h>
 #include <odp_align_internal.h>
@@ -17,51 +18,35 @@
 #include <odp_config.h>
 #include <odp_hints.h>
 #include <odp_debug_internal.h>
+#include <odp_atomic_internal.h>
 
 #include <string.h>
 #include <stdlib.h>
 
 
-#ifdef POOL_USE_TICKETLOCK
-#include <odp_ticketlock.h>
-#define LOCK(a)      odp_ticketlock_lock(a)
-#define UNLOCK(a)    odp_ticketlock_unlock(a)
-#define LOCK_INIT(a) odp_ticketlock_init(a)
-#else
-#include <odp_spinlock.h>
-#define LOCK(a)      odp_spinlock_lock(a)
-#define UNLOCK(a)    odp_spinlock_unlock(a)
-#define LOCK_INIT(a) odp_spinlock_init(a)
-#endif
-
-
 #if ODP_CONFIG_BUFFER_POOLS > ODP_BUFFER_MAX_POOLS
 #error ODP_CONFIG_BUFFER_POOLS > ODP_BUFFER_MAX_POOLS
 #endif
 
-#define NULL_INDEX ((uint32_t)-1)
 
-union buffer_type_any_u {
+typedef union buffer_type_any_u {
        odp_buffer_hdr_t  buf;
        odp_packet_hdr_t  pkt;
        odp_timeout_hdr_t tmo;
-};
-
-ODP_STATIC_ASSERT((sizeof(union buffer_type_any_u) % 8) == 0,
-          "BUFFER_TYPE_ANY_U__SIZE_ERR");
+} odp_anybuf_t;
 
 /* Any buffer type header */
 typedef struct {
        union buffer_type_any_u any_hdr;    /* any buffer type */
-       uint8_t                 buf_data[]; /* start of buffer data area */
 } odp_any_buffer_hdr_t;
 
-
+typedef struct odp_any_hdr_stride {
+       uint8_t pad[ODP_CACHE_LINE_SIZE_ROUNDUP(sizeof(odp_any_buffer_hdr_t))];
+} odp_any_hdr_stride;
 
 
 typedef struct pool_table_t {
        pool_entry_t pool[ODP_CONFIG_BUFFER_POOLS];
-
 } pool_table_t;
 
 
@@ -71,34 +56,8 @@ static pool_table_t *pool_tbl;
 /* Pool entry pointers (for inlining) */
 void *pool_entry_ptr[ODP_CONFIG_BUFFER_POOLS];
 
-
-static __thread odp_buffer_chunk_hdr_t *local_chunk[ODP_CONFIG_BUFFER_POOLS];
-
-
-static inline odp_buffer_pool_t pool_index_to_handle(uint32_t pool_id)
-{
-       return pool_id + 1;
-}
-
-
-
-
-static inline void set_handle(odp_buffer_hdr_t *hdr,
-                             pool_entry_t *pool, uint32_t index)
-{
-       odp_buffer_pool_t pool_hdl = pool->s.pool_hdl;
-       uint32_t          pool_id  = pool_handle_to_index(pool_hdl);
-
-       if (pool_id >= ODP_CONFIG_BUFFER_POOLS)
-               ODP_ABORT("set_handle: Bad pool handle %u\n", pool_hdl);
-
-       if (index > ODP_BUFFER_MAX_INDEX)
-               ODP_ERR("set_handle: Bad buffer index\n");
-
-       hdr->handle.pool_id = pool_id;
-       hdr->handle.index   = index;
-}
-
+/* Local cache for buffer alloc/free acceleration */
+static __thread local_cache_t local_cache[ODP_CONFIG_BUFFER_POOLS];
 
 int odp_buffer_pool_init_global(void)
 {
@@ -119,9 +78,9 @@ int odp_buffer_pool_init_global(void)
        for (i = 0; i < ODP_CONFIG_BUFFER_POOLS; i++) {
                /* init locks */
                pool_entry_t *pool = &pool_tbl->pool[i];
-               LOCK_INIT(&pool->s.lock);
+               POOL_LOCK_INIT(&pool->s.lock);
                pool->s.pool_hdl = pool_index_to_handle(i);
-
+               pool->s.pool_id = i;
                pool_entry_ptr[i] = pool;
        }
 
@@ -133,269 +92,278 @@ int odp_buffer_pool_init_global(void)
        return 0;
 }
 
+/**
+ * Buffer pool creation
+ */
 
-static odp_buffer_hdr_t *index_to_hdr(pool_entry_t *pool, uint32_t index)
-{
-       odp_buffer_hdr_t *hdr;
-
-       hdr = (odp_buffer_hdr_t *)(pool->s.buf_base + index * pool->s.buf_size);
-       return hdr;
-}
-
-
-static void add_buf_index(odp_buffer_chunk_hdr_t *chunk_hdr, uint32_t index)
-{
-       uint32_t i = chunk_hdr->chunk.num_bufs;
-       chunk_hdr->chunk.buf_index[i] = index;
-       chunk_hdr->chunk.num_bufs++;
-}
-
-
-static uint32_t rem_buf_index(odp_buffer_chunk_hdr_t *chunk_hdr)
-{
-       uint32_t index;
-       uint32_t i;
-
-       i = chunk_hdr->chunk.num_bufs - 1;
-       index = chunk_hdr->chunk.buf_index[i];
-       chunk_hdr->chunk.num_bufs--;
-       return index;
-}
-
-
-static odp_buffer_chunk_hdr_t *next_chunk(pool_entry_t *pool,
-                                         odp_buffer_chunk_hdr_t *chunk_hdr)
-{
-       uint32_t index;
-
-       index = chunk_hdr->chunk.buf_index[ODP_BUFS_PER_CHUNK-1];
-       if (index == NULL_INDEX)
-               return NULL;
-       else
-               return (odp_buffer_chunk_hdr_t *)index_to_hdr(pool, index);
-}
-
-
-static odp_buffer_chunk_hdr_t *rem_chunk(pool_entry_t *pool)
+odp_buffer_pool_t odp_buffer_pool_create(const char *name,
+                                        odp_shm_t shm,
+                                        odp_buffer_pool_param_t *params)
 {
-       odp_buffer_chunk_hdr_t *chunk_hdr;
-
-       chunk_hdr = pool->s.head;
-       if (chunk_hdr == NULL) {
-               /* Pool is empty */
-               return NULL;
-       }
-
-       pool->s.head = next_chunk(pool, chunk_hdr);
-       pool->s.free_bufs -= ODP_BUFS_PER_CHUNK;
+       odp_buffer_pool_t pool_hdl = ODP_BUFFER_POOL_INVALID;
+       pool_entry_t *pool;
+       uint32_t i, headroom = 0, tailroom = 0;
 
-       /* unlink */
-       rem_buf_index(chunk_hdr);
-       return chunk_hdr;
-}
+       /* Default initialization paramters */
+       static _odp_buffer_pool_init_t default_init_params = {
+               .udata_size = 0,
+               .buf_init = NULL,
+               .buf_init_arg = NULL,
+       };
 
+       _odp_buffer_pool_init_t *init_params = &default_init_params;
 
-static void add_chunk(pool_entry_t *pool, odp_buffer_chunk_hdr_t *chunk_hdr)
-{
-       if (pool->s.head) /* link pool head to the chunk */
-               add_buf_index(chunk_hdr, pool->s.head->buf_hdr.index);
-       else
-               add_buf_index(chunk_hdr, NULL_INDEX);
+       if (params == NULL)
+               return ODP_BUFFER_POOL_INVALID;
 
-       pool->s.head = chunk_hdr;
-       pool->s.free_bufs += ODP_BUFS_PER_CHUNK;
-}
+       /* Restriction for v1.0: All buffers are unsegmented */
+       const int unsegmented = 1;
 
+       /* Restriction for v1.0: No zeroization support */
+       const int zeroized = 0;
 
-static void check_align(pool_entry_t *pool, odp_buffer_hdr_t *hdr)
-{
-       if (!ODP_ALIGNED_CHECK_POWER_2(hdr->addr, pool->s.user_align)) {
-               ODP_ABORT("check_align: user data align error %p, align %zu\n",
-                         hdr->addr, pool->s.user_align);
-       }
+       /* Restriction for v1.0: No udata support */
+       uint32_t udata_stride = (init_params->udata_size > sizeof(void *)) ?
+               ODP_CACHE_LINE_SIZE_ROUNDUP(init_params->udata_size) :
+               0;
 
-       if (!ODP_ALIGNED_CHECK_POWER_2(hdr, ODP_CACHE_LINE_SIZE)) {
-               ODP_ABORT("check_align: hdr align error %p, align %i\n",
-                         hdr, ODP_CACHE_LINE_SIZE);
-       }
-}
+       uint32_t blk_size, buf_stride;
+       uint32_t buf_align = params->buf_align;
 
+       /* Validate requested buffer alignment */
+       if (buf_align > ODP_CONFIG_BUFFER_ALIGN_MAX ||
+           buf_align != ODP_ALIGN_ROUNDDOWN_POWER_2(buf_align, buf_align))
+               return ODP_BUFFER_POOL_INVALID;
 
-static void fill_hdr(void *ptr, pool_entry_t *pool, uint32_t index,
-                    int buf_type)
-{
-       odp_buffer_hdr_t *hdr = (odp_buffer_hdr_t *)ptr;
-       size_t size = pool->s.hdr_size;
-       uint8_t *buf_data;
+       /* Set correct alignment based on input request */
+       if (buf_align == 0)
+               buf_align = ODP_CACHE_LINE_SIZE;
+       else if (buf_align < ODP_CONFIG_BUFFER_ALIGN_MIN)
+               buf_align = ODP_CONFIG_BUFFER_ALIGN_MIN;
 
-       if (buf_type == ODP_BUFFER_TYPE_CHUNK)
-               size = sizeof(odp_buffer_chunk_hdr_t);
+       /* Calculate space needed for buffer blocks and metadata */
+       switch (params->buf_type) {
+       case ODP_BUFFER_TYPE_RAW:
+       case ODP_BUFFER_TYPE_TIMEOUT:
+               blk_size = params->buf_size;
 
-       switch (pool->s.buf_type) {
-               odp_raw_buffer_hdr_t *raw_hdr;
-               odp_packet_hdr_t *packet_hdr;
-               odp_timeout_hdr_t *tmo_hdr;
-               odp_any_buffer_hdr_t *any_hdr;
+               /* Optimize small raw buffers */
+               if (blk_size > ODP_MAX_INLINE_BUF || params->buf_align != 0)
+                       blk_size = ODP_ALIGN_ROUNDUP(blk_size, buf_align);
 
-       case ODP_BUFFER_TYPE_RAW:
-               raw_hdr  = ptr;
-               buf_data = raw_hdr->buf_data;
+               buf_stride = params->buf_type == ODP_BUFFER_TYPE_RAW ?
+                       sizeof(odp_buffer_hdr_stride) :
+                       sizeof(odp_timeout_hdr_stride);
                break;
+
        case ODP_BUFFER_TYPE_PACKET:
-               packet_hdr = ptr;
-               buf_data   = packet_hdr->buf_data;
-               break;
-       case ODP_BUFFER_TYPE_TIMEOUT:
-               tmo_hdr  = ptr;
-               buf_data = tmo_hdr->buf_data;
-               break;
        case ODP_BUFFER_TYPE_ANY:
-               any_hdr  = ptr;
-               buf_data = any_hdr->buf_data;
+               headroom = ODP_CONFIG_PACKET_HEADROOM;
+               tailroom = ODP_CONFIG_PACKET_TAILROOM;
+               if (unsegmented)
+                       blk_size = ODP_ALIGN_ROUNDUP(
+                               headroom + params->buf_size + tailroom,
+                               buf_align);
+               else
+                       blk_size = ODP_ALIGN_ROUNDUP(
+                               headroom + params->buf_size + tailroom,
+                               ODP_CONFIG_PACKET_BUF_LEN_MIN);
+               buf_stride = params->buf_type == ODP_BUFFER_TYPE_PACKET ?
+                       sizeof(odp_packet_hdr_stride) :
+                       sizeof(odp_any_hdr_stride);
                break;
-       default:
-               ODP_ABORT("Bad buffer type\n");
-       }
-
-       memset(hdr, 0, size);
-
-       set_handle(hdr, pool, index);
-
-       hdr->addr     = &buf_data[pool->s.buf_offset - pool->s.hdr_size];
-       hdr->index    = index;
-       hdr->size     = pool->s.user_size;
-       hdr->pool_hdl = pool->s.pool_hdl;
-       hdr->type     = buf_type;
-
-       check_align(pool, hdr);
-}
-
-
-static void link_bufs(pool_entry_t *pool)
-{
-       odp_buffer_chunk_hdr_t *chunk_hdr;
-       size_t hdr_size;
-       size_t data_size;
-       size_t data_align;
-       size_t tot_size;
-       size_t offset;
-       size_t min_size;
-       uint64_t pool_size;
-       uintptr_t buf_base;
-       uint32_t index;
-       uintptr_t pool_base;
-       int buf_type;
-
-       buf_type   = pool->s.buf_type;
-       data_size  = pool->s.user_size;
-       data_align = pool->s.user_align;
-       pool_size  = pool->s.pool_size;
-       pool_base  = (uintptr_t) pool->s.pool_base_addr;
-
-       if (buf_type == ODP_BUFFER_TYPE_RAW) {
-               hdr_size = sizeof(odp_raw_buffer_hdr_t);
-       } else if (buf_type == ODP_BUFFER_TYPE_PACKET) {
-               hdr_size = sizeof(odp_packet_hdr_t);
-       } else if (buf_type == ODP_BUFFER_TYPE_TIMEOUT) {
-               hdr_size = sizeof(odp_timeout_hdr_t);
-       } else if (buf_type == ODP_BUFFER_TYPE_ANY) {
-               hdr_size = sizeof(odp_any_buffer_hdr_t);
-       } else
-               ODP_ABORT("odp_buffer_pool_create: Bad type %i\n", buf_type);
-
-
-       /* Chunk must fit into buffer data area.*/
-       min_size = sizeof(odp_buffer_chunk_hdr_t) - hdr_size;
-       if (data_size < min_size)
-               data_size = min_size;
-
-       /* Roundup data size to full cachelines */
-       data_size = ODP_CACHE_LINE_SIZE_ROUNDUP(data_size);
-
-       /* Min cacheline alignment for buffer header and data */
-       data_align = ODP_CACHE_LINE_SIZE_ROUNDUP(data_align);
-       offset     = ODP_CACHE_LINE_SIZE_ROUNDUP(hdr_size);
-
-       /* Multiples of cacheline size */
-       if (data_size > data_align)
-               tot_size = data_size + offset;
-       else
-               tot_size = data_align + offset;
-
-       /* First buffer */
-       buf_base = ODP_ALIGN_ROUNDUP(pool_base + offset, data_align) - offset;
-
-       pool->s.hdr_size   = hdr_size;
-       pool->s.buf_base   = buf_base;
-       pool->s.buf_size   = tot_size;
-       pool->s.buf_offset = offset;
-       index = 0;
-
-       chunk_hdr = (odp_buffer_chunk_hdr_t *)index_to_hdr(pool, index);
-       pool->s.head   = NULL;
-       pool_size     -= buf_base - pool_base;
-
-       while (pool_size > ODP_BUFS_PER_CHUNK * tot_size) {
-               int i;
-
-               fill_hdr(chunk_hdr, pool, index, ODP_BUFFER_TYPE_CHUNK);
 
-               index++;
-
-               for (i = 0; i < ODP_BUFS_PER_CHUNK - 1; i++) {
-                       odp_buffer_hdr_t *hdr = index_to_hdr(pool, index);
-
-                       fill_hdr(hdr, pool, index, buf_type);
-
-                       add_buf_index(chunk_hdr, index);
-                       index++;
-               }
-
-               add_chunk(pool, chunk_hdr);
-
-               chunk_hdr = (odp_buffer_chunk_hdr_t *)index_to_hdr(pool,
-                                                                  index);
-               pool->s.num_bufs += ODP_BUFS_PER_CHUNK;
-               pool_size -=  ODP_BUFS_PER_CHUNK * tot_size;
+       default:
+               return ODP_BUFFER_POOL_INVALID;
        }
-}
 
+       /* Validate requested number of buffers against addressable limits */
+       if (params->num_bufs >
+           (ODP_BUFFER_MAX_BUFFERS / (buf_stride / ODP_CACHE_LINE_SIZE)))
+               return ODP_BUFFER_POOL_INVALID;
 
-odp_buffer_pool_t odp_buffer_pool_create(const char *name,
-                                        void *base_addr, uint64_t size,
-                                        size_t buf_size, size_t buf_align,
-                                        int buf_type)
-{
-       odp_buffer_pool_t pool_hdl = ODP_BUFFER_POOL_INVALID;
-       pool_entry_t *pool;
-       uint32_t i;
-
+       /* Find an unused buffer pool slot and iniitalize it as requested */
        for (i = 0; i < ODP_CONFIG_BUFFER_POOLS; i++) {
                pool = get_pool_entry(i);
 
-               LOCK(&pool->s.lock);
+               POOL_LOCK(&pool->s.lock);
+               if (pool->s.pool_shm != ODP_SHM_INVALID) {
+                       POOL_UNLOCK(&pool->s.lock);
+                       continue;
+               }
+
+               /* found free pool */
+               size_t block_size, pad_size, mdata_size, udata_size;
 
-               if (pool->s.buf_base == 0) {
-                       /* found free pool */
+               pool->s.flags.all = 0;
 
+               if (name == NULL) {
+                       pool->s.name[0] = 0;
+               } else {
                        strncpy(pool->s.name, name,
                                ODP_BUFFER_POOL_NAME_LEN - 1);
                        pool->s.name[ODP_BUFFER_POOL_NAME_LEN - 1] = 0;
-                       pool->s.pool_base_addr = base_addr;
-                       pool->s.pool_size      = size;
-                       pool->s.user_size      = buf_size;
-                       pool->s.user_align     = buf_align;
-                       pool->s.buf_type       = buf_type;
-
-                       link_bufs(pool);
+                       pool->s.flags.has_name = 1;
+               }
 
-                       UNLOCK(&pool->s.lock);
+               pool->s.params = *params;
+               pool->s.init_params = *init_params;
+               pool->s.buf_align = buf_align;
+
+               /* Optimize for short buffers: Data stored in buffer hdr */
+               if (blk_size <= ODP_MAX_INLINE_BUF) {
+                       block_size = 0;
+                       pool->s.buf_align = blk_size == 0 ? 0 : sizeof(void *);
+               } else {
+                       block_size = params->num_bufs * blk_size;
+                       pool->s.buf_align = buf_align;
+               }
 
-                       pool_hdl = pool->s.pool_hdl;
-                       break;
+               pad_size = ODP_CACHE_LINE_SIZE_ROUNDUP(block_size) - block_size;
+               mdata_size = params->num_bufs * buf_stride;
+               udata_size = params->num_bufs * udata_stride;
+
+               pool->s.pool_size = ODP_PAGE_SIZE_ROUNDUP(block_size +
+                                                         pad_size +
+                                                         mdata_size +
+                                                         udata_size);
+
+               if (shm == ODP_SHM_NULL) {
+                       shm = odp_shm_reserve(pool->s.name,
+                                             pool->s.pool_size,
+                                             ODP_PAGE_SIZE, 0);
+                       if (shm == ODP_SHM_INVALID) {
+                               POOL_UNLOCK(&pool->s.lock);
+                               return ODP_BUFFER_INVALID;
+                       }
+                       pool->s.pool_base_addr = odp_shm_addr(shm);
+               } else {
+                       odp_shm_info_t info;
+                       if (odp_shm_info(shm, &info) != 0 ||
+                           info.size < pool->s.pool_size) {
+                               POOL_UNLOCK(&pool->s.lock);
+                               return ODP_BUFFER_POOL_INVALID;
+                       }
+                       pool->s.pool_base_addr = odp_shm_addr(shm);
+                       void *page_addr =
+                               ODP_ALIGN_ROUNDUP_PTR(pool->s.pool_base_addr,
+                                                     ODP_PAGE_SIZE);
+                       if (pool->s.pool_base_addr != page_addr) {
+                               if (info.size < pool->s.pool_size +
+                                   ((size_t)page_addr -
+                                    (size_t)pool->s.pool_base_addr)) {
+                                       POOL_UNLOCK(&pool->s.lock);
+                                       return ODP_BUFFER_POOL_INVALID;
+                               }
+                               pool->s.pool_base_addr = page_addr;
+                       }
+                       pool->s.flags.user_supplied_shm = 1;
                }
 
-               UNLOCK(&pool->s.lock);
+               pool->s.pool_shm = shm;
+
+               /* Now safe to unlock since pool entry has been allocated */
+               POOL_UNLOCK(&pool->s.lock);
+
+               pool->s.flags.unsegmented = unsegmented;
+               pool->s.flags.zeroized = zeroized;
+               pool->s.seg_size = unsegmented ?
+                       blk_size : ODP_CONFIG_PACKET_BUF_LEN_MIN;
+
+
+               uint8_t *block_base_addr = pool->s.pool_base_addr;
+               uint8_t *mdata_base_addr =
+                       block_base_addr + block_size + pad_size;
+               uint8_t *udata_base_addr = mdata_base_addr + mdata_size;
+
+               /* Pool mdata addr is used for indexing buffer metadata */
+               pool->s.pool_mdata_addr = mdata_base_addr;
+
+               pool->s.buf_stride = buf_stride;
+               _odp_atomic_ptr_store(&pool->s.buf_freelist, NULL,
+                                     _ODP_MEMMODEL_RLX);
+               _odp_atomic_ptr_store(&pool->s.blk_freelist, NULL,
+                                     _ODP_MEMMODEL_RLX);
+
+               /* Initialization will increment these to their target vals */
+               odp_atomic_store_u32(&pool->s.bufcount, 0);
+               odp_atomic_store_u32(&pool->s.blkcount, 0);
+
+               uint8_t *buf = udata_base_addr - buf_stride;
+               uint8_t *udat = udata_stride == 0 ? NULL :
+                       block_base_addr - udata_stride;
+
+               /* Init buffer common header and add to pool buffer freelist */
+               do {
+                       odp_buffer_hdr_t *tmp =
+                               (odp_buffer_hdr_t *)(void *)buf;
+
+                       /* Iniitalize buffer metadata */
+                       tmp->allocator = ODP_FREEBUF;
+                       tmp->flags.all = 0;
+                       tmp->flags.zeroized = zeroized;
+                       tmp->size = 0;
+                       odp_atomic_store_u32(&tmp->ref_count, 0);
+                       tmp->type = params->buf_type;
+                       tmp->pool_hdl = pool->s.pool_hdl;
+                       tmp->udata_addr = (void *)udat;
+                       tmp->udata_size = init_params->udata_size;
+                       tmp->segcount = 0;
+                       tmp->segsize = pool->s.seg_size;
+                       tmp->handle.handle = odp_buffer_encode_handle(tmp);
+
+                       /* Set 1st seg addr for zero-len buffers */
+                       tmp->addr[0] = NULL;
+
+                       /* Special case for short buffer data */
+                       if (blk_size <= ODP_MAX_INLINE_BUF) {
+                               tmp->flags.hdrdata = 1;
+                               if (blk_size > 0) {
+                                       tmp->segcount = 1;
+                                       tmp->addr[0] = &tmp->addr[1];
+                                       tmp->size = blk_size;
+                               }
+                       }
+
+                       /* Push buffer onto pool's freelist */
+                       ret_buf(&pool->s, tmp);
+                       buf  -= buf_stride;
+                       udat -= udata_stride;
+               } while (buf >= mdata_base_addr);
+
+               /* Form block freelist for pool */
+               uint8_t *blk =
+                       block_base_addr + block_size - pool->s.seg_size;
+
+               if (blk_size > ODP_MAX_INLINE_BUF)
+                       do {
+                               ret_blk(&pool->s, blk);
+                               blk -= pool->s.seg_size;
+                       } while (blk >= block_base_addr);
+
+               /* Initialize pool statistics counters */
+               odp_atomic_store_u64(&pool->s.bufallocs, 0);
+               odp_atomic_store_u64(&pool->s.buffrees, 0);
+               odp_atomic_store_u64(&pool->s.blkallocs, 0);
+               odp_atomic_store_u64(&pool->s.blkfrees, 0);
+               odp_atomic_store_u64(&pool->s.bufempty, 0);
+               odp_atomic_store_u64(&pool->s.blkempty, 0);
+               odp_atomic_store_u64(&pool->s.high_wm_count, 0);
+               odp_atomic_store_u64(&pool->s.low_wm_count, 0);
+
+               /* Reset other pool globals to initial state */
+               pool->s.low_wm_assert = 0;
+               pool->s.quiesced = 0;
+               pool->s.low_wm_assert = 0;
+               pool->s.headroom = headroom;
+               pool->s.tailroom = tailroom;
+
+               /* Watermarks are hard-coded for now to control caching */
+               pool->s.high_wm = params->num_bufs / 2;
+               pool->s.low_wm = params->num_bufs / 4;
+
+               pool_hdl = pool->s.pool_hdl;
+               break;
        }
 
        return pool_hdl;
@@ -410,157 +378,171 @@ odp_buffer_pool_t odp_buffer_pool_lookup(const char 
*name)
        for (i = 0; i < ODP_CONFIG_BUFFER_POOLS; i++) {
                pool = get_pool_entry(i);
 
-               LOCK(&pool->s.lock);
+               POOL_LOCK(&pool->s.lock);
                if (strcmp(name, pool->s.name) == 0) {
                        /* found it */
-                       UNLOCK(&pool->s.lock);
+                       POOL_UNLOCK(&pool->s.lock);
                        return pool->s.pool_hdl;
                }
-               UNLOCK(&pool->s.lock);
+               POOL_UNLOCK(&pool->s.lock);
        }
 
        return ODP_BUFFER_POOL_INVALID;
 }
 
 
-odp_buffer_t odp_buffer_alloc(odp_buffer_pool_t pool_hdl)
+odp_buffer_t buffer_alloc(odp_buffer_pool_t pool_hdl, size_t size)
 {
-       pool_entry_t *pool;
-       odp_buffer_chunk_hdr_t *chunk;
-       odp_buffer_bits_t handle;
        uint32_t pool_id = pool_handle_to_index(pool_hdl);
+       pool_entry_t *pool = get_pool_entry(pool_id);
+       uintmax_t totsize = pool->s.headroom + size + pool->s.tailroom;
+       odp_anybuf_t *buf;
 
-       pool  = get_pool_entry(pool_id);
-       chunk = local_chunk[pool_id];
+       /* Reject oversized allocation requests */
+       if ((pool->s.flags.unsegmented && totsize > pool->s.seg_size) ||
+           (!pool->s.flags.unsegmented &&
+            totsize > ODP_CONFIG_PACKET_BUF_LEN_MAX))
+               return ODP_BUFFER_INVALID;
 
-       if (chunk == NULL) {
-               LOCK(&pool->s.lock);
-               chunk = rem_chunk(pool);
-               UNLOCK(&pool->s.lock);
+       /* Try to satisfy request from the local cache */
+       buf = (odp_anybuf_t *)(void *)get_local_buf(&local_cache[pool_id],
+                                                   &pool->s, totsize);
 
-               if (chunk == NULL)
+       /* If cache is empty, satisfy request from the pool */
+       if (odp_unlikely(buf == NULL)) {
+               buf = (odp_anybuf_t *)(void *)get_buf(&pool->s);
+
+               if (odp_unlikely(buf == NULL))
                        return ODP_BUFFER_INVALID;
 
-               local_chunk[pool_id] = chunk;
+               /* Get blocks for this buffer, if pool uses application data */
+               if (buf->buf.size < totsize) {
+                       intmax_t needed = totsize - buf->buf.size;
+                       do {
+                               uint8_t *blk = get_blk(&pool->s);
+                               if (blk == NULL) {
+                                       ret_buf(&pool->s, &buf->buf);
+                                       return ODP_BUFFER_INVALID;
+                               }
+                               buf->buf.addr[buf->buf.segcount++] = blk;
+                               needed -= pool->s.seg_size;
+                       } while (needed > 0);
+                       buf->buf.size = buf->buf.segcount * pool->s.seg_size;
+               }
        }
 
-       if (chunk->chunk.num_bufs == 0) {
-               /* give the chunk buffer */
-               local_chunk[pool_id] = NULL;
-               chunk->buf_hdr.type = pool->s.buf_type;
+       /* By default, buffers inherit their pool's zeroization setting */
+       buf->buf.flags.zeroized = pool->s.flags.zeroized;
 
-               handle = chunk->buf_hdr.handle;
-       } else {
-               odp_buffer_hdr_t *hdr;
-               uint32_t index;
-               index = rem_buf_index(chunk);
-               hdr = index_to_hdr(pool, index);
+       if (buf->buf.type == ODP_BUFFER_TYPE_PACKET) {
+               packet_init(pool, &buf->pkt, size);
 
-               handle = hdr->handle;
+               if (pool->s.init_params.buf_init != NULL)
+                       (*pool->s.init_params.buf_init)
+                               (buf->buf.handle.handle,
+                                pool->s.init_params.buf_init_arg);
        }
 
-       return handle.u32;
+       return odp_hdr_to_buf(&buf->buf);
 }
 
+odp_buffer_t odp_buffer_alloc(odp_buffer_pool_t pool_hdl)
+{
+       return buffer_alloc(pool_hdl,
+                           odp_pool_to_entry(pool_hdl)->s.params.buf_size);
+}
 
 void odp_buffer_free(odp_buffer_t buf)
 {
-       odp_buffer_hdr_t *hdr;
-       uint32_t pool_id;
-       pool_entry_t *pool;
-       odp_buffer_chunk_hdr_t *chunk_hdr;
-
-       hdr       = odp_buf_to_hdr(buf);
-       pool_id   = pool_handle_to_index(hdr->pool_hdl);
-       pool      = get_pool_entry(pool_id);
-       chunk_hdr = local_chunk[pool_id];
-
-       if (chunk_hdr && chunk_hdr->chunk.num_bufs == ODP_BUFS_PER_CHUNK - 1) {
-               /* Current chunk is full. Push back to the pool */
-               LOCK(&pool->s.lock);
-               add_chunk(pool, chunk_hdr);
-               UNLOCK(&pool->s.lock);
-               chunk_hdr = NULL;
-       }
+       odp_buffer_hdr_t *buf_hdr = odp_buf_to_hdr(buf);
+       pool_entry_t *pool = odp_buf_to_pool(buf_hdr);
 
-       if (chunk_hdr == NULL) {
-               /* Use this buffer */
-               chunk_hdr = (odp_buffer_chunk_hdr_t *)hdr;
-               local_chunk[pool_id] = chunk_hdr;
-               chunk_hdr->chunk.num_bufs = 0;
-       } else {
-               /* Add to current chunk */
-               add_buf_index(chunk_hdr, hdr->index);
-       }
+       if (odp_unlikely(pool->s.low_wm_assert))
+               ret_buf(&pool->s, buf_hdr);
+       else
+               ret_local_buf(&local_cache[pool->s.pool_id], buf_hdr);
 }
 
-
-odp_buffer_pool_t odp_buffer_pool(odp_buffer_t buf)
+void _odp_flush_caches(void)
 {
-       odp_buffer_hdr_t *hdr;
+       int i;
 
-       hdr = odp_buf_to_hdr(buf);
-       return hdr->pool_hdl;
+       for (i = 0; i < ODP_CONFIG_BUFFER_POOLS; i++) {
+               pool_entry_t *pool = get_pool_entry(i);
+               flush_cache(&local_cache[i], &pool->s);
+       }
 }
 
-
 void odp_buffer_pool_print(odp_buffer_pool_t pool_hdl)
 {
        pool_entry_t *pool;
-       odp_buffer_chunk_hdr_t *chunk_hdr;
-       uint32_t i;
        uint32_t pool_id;
 
        pool_id = pool_handle_to_index(pool_hdl);
        pool    = get_pool_entry(pool_id);
 
-       ODP_PRINT("Pool info\n");
-       ODP_PRINT("---------\n");
-       ODP_PRINT("  pool          %i\n",           pool->s.pool_hdl);
-       ODP_PRINT("  name          %s\n",           pool->s.name);
-       ODP_PRINT("  pool base     %p\n",           pool->s.pool_base_addr);
-       ODP_PRINT("  buf base      0x%"PRIxPTR"\n", pool->s.buf_base);
-       ODP_PRINT("  pool size     0x%"PRIx64"\n",  pool->s.pool_size);
-       ODP_PRINT("  buf size      %zu\n",          pool->s.user_size);
-       ODP_PRINT("  buf align     %zu\n",          pool->s.user_align);
-       ODP_PRINT("  hdr size      %zu\n",          pool->s.hdr_size);
-       ODP_PRINT("  alloc size    %zu\n",          pool->s.buf_size);
-       ODP_PRINT("  offset to hdr %zu\n",          pool->s.buf_offset);
-       ODP_PRINT("  num bufs      %"PRIu64"\n",    pool->s.num_bufs);
-       ODP_PRINT("  free bufs     %"PRIu64"\n",    pool->s.free_bufs);
-
-       /* first chunk */
-       chunk_hdr = pool->s.head;
-
-       if (chunk_hdr == NULL) {
-               ODP_ERR("  POOL EMPTY\n");
-               return;
-       }
-
-       ODP_PRINT("\n  First chunk\n");
-
-       for (i = 0; i < chunk_hdr->chunk.num_bufs - 1; i++) {
-               uint32_t index;
-               odp_buffer_hdr_t *hdr;
-
-               index = chunk_hdr->chunk.buf_index[i];
-               hdr   = index_to_hdr(pool, index);
-
-               ODP_PRINT("  [%i] addr %p, id %"PRIu32"\n", i, hdr->addr,
-                         index);
-       }
-
-       ODP_PRINT("  [%i] addr %p, id %"PRIu32"\n", i, chunk_hdr->buf_hdr.addr,
-                 chunk_hdr->buf_hdr.index);
-
-       /* next chunk */
-       chunk_hdr = next_chunk(pool, chunk_hdr);
+       uint32_t bufcount  = odp_atomic_load_u32(&pool->s.bufcount);
+       uint32_t blkcount  = odp_atomic_load_u32(&pool->s.blkcount);
+       uint64_t bufallocs = odp_atomic_load_u64(&pool->s.bufallocs);
+       uint64_t buffrees  = odp_atomic_load_u64(&pool->s.buffrees);
+       uint64_t blkallocs = odp_atomic_load_u64(&pool->s.blkallocs);
+       uint64_t blkfrees  = odp_atomic_load_u64(&pool->s.blkfrees);
+       uint64_t bufempty  = odp_atomic_load_u64(&pool->s.bufempty);
+       uint64_t blkempty  = odp_atomic_load_u64(&pool->s.blkempty);
+       uint64_t hiwmct    = odp_atomic_load_u64(&pool->s.high_wm_count);
+       uint64_t lowmct    = odp_atomic_load_u64(&pool->s.low_wm_count);
+
+       ODP_DBG("Pool info\n");
+       ODP_DBG("---------\n");
+       ODP_DBG(" pool            %i\n", pool->s.pool_hdl);
+       ODP_DBG(" name            %s\n",
+               pool->s.flags.has_name ? pool->s.name : "Unnamed Pool");
+       ODP_DBG(" pool type       %s\n",
+               pool->s.params.buf_type == ODP_BUFFER_TYPE_RAW ? "raw" :
+              (pool->s.params.buf_type == ODP_BUFFER_TYPE_PACKET ? "packet" :
+              (pool->s.params.buf_type == ODP_BUFFER_TYPE_TIMEOUT ? "timeout" :
+              (pool->s.params.buf_type == ODP_BUFFER_TYPE_ANY ? "any" :
+               "unknown"))));
+       ODP_DBG(" pool storage    %sODP managed\n",
+               pool->s.flags.user_supplied_shm ?
+               "application provided, " : "");
+       ODP_DBG(" pool status     %s\n",
+               pool->s.quiesced ? "quiesced" : "active");
+       ODP_DBG(" pool opts       %s, %s, %s\n",
+               pool->s.flags.unsegmented ? "unsegmented" : "segmented",
+               pool->s.flags.zeroized ? "zeroized" : "non-zeroized",
+               pool->s.flags.predefined  ? "predefined" : "created");
+       ODP_DBG(" pool base       %p\n",  pool->s.pool_base_addr);
+       ODP_DBG(" pool size       %zu (%zu pages)\n",
+               pool->s.pool_size, pool->s.pool_size / ODP_PAGE_SIZE);
+       ODP_DBG(" pool mdata base %p\n",  pool->s.pool_mdata_addr);
+       ODP_DBG(" udata size      %zu\n", pool->s.init_params.udata_size);
+       ODP_DBG(" headroom        %u\n",  pool->s.headroom);
+       ODP_DBG(" buf size        %zu\n", pool->s.params.buf_size);
+       ODP_DBG(" tailroom        %u\n",  pool->s.tailroom);
+       ODP_DBG(" buf align       %u requested, %u used\n",
+               pool->s.params.buf_align, pool->s.buf_align);
+       ODP_DBG(" num bufs        %u\n",  pool->s.params.num_bufs);
+       ODP_DBG(" bufs available  %u %s\n", bufcount,
+               pool->s.low_wm_assert ? " **low wm asserted**" : "");
+       ODP_DBG(" bufs in use     %u\n",  pool->s.params.num_bufs - bufcount);
+       ODP_DBG(" buf allocs      %lu\n", bufallocs);
+       ODP_DBG(" buf frees       %lu\n", buffrees);
+       ODP_DBG(" buf empty       %lu\n", bufempty);
+       ODP_DBG(" blk size        %zu\n",
+               pool->s.seg_size > ODP_MAX_INLINE_BUF ? pool->s.seg_size : 0);
+       ODP_DBG(" blks available  %u\n",  blkcount);
+       ODP_DBG(" blk allocs      %lu\n", blkallocs);
+       ODP_DBG(" blk frees       %lu\n", blkfrees);
+       ODP_DBG(" blk empty       %lu\n", blkempty);
+       ODP_DBG(" high wm value   %lu\n", pool->s.high_wm);
+       ODP_DBG(" high wm count   %lu\n", hiwmct);
+       ODP_DBG(" low wm value    %lu\n", pool->s.low_wm);
+       ODP_DBG(" low wm count    %lu\n", lowmct);
+}
 
-       if (chunk_hdr) {
-               ODP_PRINT("  Next chunk\n");
-               ODP_PRINT("  addr %p, id %"PRIu32"\n", chunk_hdr->buf_hdr.addr,
-                         chunk_hdr->buf_hdr.index);
-       }
 
-       ODP_PRINT("\n");
+odp_buffer_pool_t odp_buffer_pool(odp_buffer_t buf)
+{
+       return odp_buf_to_hdr(buf)->pool_hdl;
 }
diff --git a/platform/linux-generic/odp_linux.c 
b/platform/linux-generic/odp_linux.c
index ecd77b3..95761a9 100644
--- a/platform/linux-generic/odp_linux.c
+++ b/platform/linux-generic/odp_linux.c
@@ -43,7 +43,9 @@ static void *odp_run_start_routine(void *arg)
                return NULL;
        }
 
-       return start_args->start_routine(start_args->arg);
+       void *ret = start_args->start_routine(start_args->arg);
+       _odp_flush_caches();
+       return ret;
 }
 
 
diff --git a/platform/linux-generic/odp_packet.c 
b/platform/linux-generic/odp_packet.c
index a1bf18e..726e086 100644
--- a/platform/linux-generic/odp_packet.c
+++ b/platform/linux-generic/odp_packet.c
@@ -24,17 +24,9 @@ static inline uint8_t parse_ipv6(odp_packet_hdr_t *pkt_hdr,
 void odp_packet_init(odp_packet_t pkt)
 {
        odp_packet_hdr_t *const pkt_hdr = odp_packet_hdr(pkt);
-       const size_t start_offset = ODP_FIELD_SIZEOF(odp_packet_hdr_t, buf_hdr);
-       uint8_t *start;
-       size_t len;
-
-       start = (uint8_t *)pkt_hdr + start_offset;
-       len = ODP_OFFSETOF(odp_packet_hdr_t, buf_data) - start_offset;
-       memset(start, 0, len);
+       pool_entry_t *pool = odp_buf_to_pool(&pkt_hdr->buf_hdr);
 
-       pkt_hdr->l2_offset = ODP_PACKET_OFFSET_INVALID;
-       pkt_hdr->l3_offset = ODP_PACKET_OFFSET_INVALID;
-       pkt_hdr->l4_offset = ODP_PACKET_OFFSET_INVALID;
+       packet_init(pool, pkt_hdr, 0);
 }
 
 odp_packet_t odp_packet_from_buffer(odp_buffer_t buf)
@@ -64,7 +56,7 @@ uint8_t *odp_packet_addr(odp_packet_t pkt)
 
 uint8_t *odp_packet_data(odp_packet_t pkt)
 {
-       return odp_packet_addr(pkt) + odp_packet_hdr(pkt)->frame_offset;
+       return odp_packet_addr(pkt) + odp_packet_hdr(pkt)->headroom;
 }
 
 
@@ -131,20 +123,13 @@ void odp_packet_set_l4_offset(odp_packet_t pkt, size_t 
offset)
 
 int odp_packet_is_segmented(odp_packet_t pkt)
 {
-       odp_buffer_hdr_t *buf_hdr = odp_buf_to_hdr((odp_buffer_t)pkt);
-
-       if (buf_hdr->scatter.num_bufs == 0)
-               return 0;
-       else
-               return 1;
+       return odp_packet_hdr(pkt)->buf_hdr.segcount > 1;
 }
 
 
 int odp_packet_seg_count(odp_packet_t pkt)
 {
-       odp_buffer_hdr_t *buf_hdr = odp_buf_to_hdr((odp_buffer_t)pkt);
-
-       return (int)buf_hdr->scatter.num_bufs + 1;
+       return odp_packet_hdr(pkt)->buf_hdr.segcount;
 }
 
 
@@ -170,7 +155,7 @@ void odp_packet_parse(odp_packet_t pkt, size_t len, size_t 
frame_offset)
        uint8_t ip_proto = 0;
 
        pkt_hdr->input_flags.eth = 1;
-       pkt_hdr->frame_offset = frame_offset;
+       pkt_hdr->l2_offset = frame_offset;
        pkt_hdr->frame_len = len;
 
        if (len > ODPH_ETH_LEN_MAX)
@@ -330,8 +315,6 @@ void odp_packet_print(odp_packet_t pkt)
        len += snprintf(&str[len], n-len,
                        "  output_flags 0x%x\n", hdr->output_flags.all);
        len += snprintf(&str[len], n-len,
-                       "  frame_offset %u\n", hdr->frame_offset);
-       len += snprintf(&str[len], n-len,
                        "  l2_offset    %u\n", hdr->l2_offset);
        len += snprintf(&str[len], n-len,
                        "  l3_offset    %u\n", hdr->l3_offset);
@@ -358,14 +341,13 @@ int odp_packet_copy(odp_packet_t pkt_dst, odp_packet_t 
pkt_src)
        if (pkt_dst == ODP_PACKET_INVALID || pkt_src == ODP_PACKET_INVALID)
                return -1;
 
-       if (pkt_hdr_dst->buf_hdr.size <
-               pkt_hdr_src->frame_len + pkt_hdr_src->frame_offset)
+       if (pkt_hdr_dst->buf_hdr.size < pkt_hdr_src->frame_len)
                return -1;
 
        /* Copy packet header */
        start_dst = (uint8_t *)pkt_hdr_dst + start_offset;
        start_src = (uint8_t *)pkt_hdr_src + start_offset;
-       len = ODP_OFFSETOF(odp_packet_hdr_t, buf_data) - start_offset;
+       len = sizeof(odp_packet_hdr_t) - start_offset;
        memcpy(start_dst, start_src, len);
 
        /* Copy frame payload */
@@ -374,13 +356,6 @@ int odp_packet_copy(odp_packet_t pkt_dst, odp_packet_t 
pkt_src)
        len = pkt_hdr_src->frame_len;
        memcpy(start_dst, start_src, len);
 
-       /* Copy useful things from the buffer header */
-       pkt_hdr_dst->buf_hdr.cur_offset = pkt_hdr_src->buf_hdr.cur_offset;
-
-       /* Create a copy of the scatter list */
-       odp_buffer_copy_scatter(odp_packet_to_buffer(pkt_dst),
-                               odp_packet_to_buffer(pkt_src));
-
        return 0;
 }
 
diff --git a/platform/linux-generic/odp_queue.c 
b/platform/linux-generic/odp_queue.c
index c278094..a7c5e42 100644
--- a/platform/linux-generic/odp_queue.c
+++ b/platform/linux-generic/odp_queue.c
@@ -11,6 +11,7 @@
 #include <odp_buffer.h>
 #include <odp_buffer_internal.h>
 #include <odp_buffer_pool_internal.h>
+#include <odp_buffer_inlines.h>
 #include <odp_internal.h>
 #include <odp_shared_memory.h>
 #include <odp_schedule_internal.h>
diff --git a/platform/linux-generic/odp_schedule.c 
b/platform/linux-generic/odp_schedule.c
index 7c09c23..aa11b7b 100644
--- a/platform/linux-generic/odp_schedule.c
+++ b/platform/linux-generic/odp_schedule.c
@@ -83,8 +83,8 @@ int odp_schedule_init_global(void)
 {
        odp_shm_t shm;
        odp_buffer_pool_t pool;
-       void *pool_base;
        int i, j;
+       odp_buffer_pool_param_t params;
 
        ODP_DBG("Schedule init ... ");
 
@@ -99,20 +99,12 @@ int odp_schedule_init_global(void)
                return -1;
        }
 
-       shm = odp_shm_reserve("odp_sched_pool",
-                             SCHED_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
+       params.buf_size  = sizeof(queue_desc_t);
+       params.buf_align = 0;
+       params.num_bufs  = SCHED_POOL_SIZE/sizeof(queue_desc_t);
+       params.buf_type  = ODP_BUFFER_TYPE_RAW;
 
-       pool_base = odp_shm_addr(shm);
-
-       if (pool_base == NULL) {
-               ODP_ERR("Schedule init: Shm reserve failed.\n");
-               return -1;
-       }
-
-       pool = odp_buffer_pool_create("odp_sched_pool", pool_base,
-                                     SCHED_POOL_SIZE, sizeof(queue_desc_t),
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_RAW);
+       pool = odp_buffer_pool_create("odp_sched_pool", ODP_SHM_NULL, &params);
 
        if (pool == ODP_BUFFER_POOL_INVALID) {
                ODP_ERR("Schedule init: Pool create failed.\n");
diff --git a/platform/linux-generic/odp_timer.c 
b/platform/linux-generic/odp_timer.c
index a4fef58..7bd6874 100644
--- a/platform/linux-generic/odp_timer.c
+++ b/platform/linux-generic/odp_timer.c
@@ -5,9 +5,10 @@
  */
 
 #include <odp_timer.h>
-#include <odp_timer_internal.h>
 #include <odp_time.h>
 #include <odp_buffer_pool_internal.h>
+#include <odp_buffer_inlines.h>
+#include <odp_timer_internal.h>
 #include <odp_internal.h>
 #include <odp_atomic.h>
 #include <odp_spinlock.h>
diff --git a/test/api_test/odp_timer_ping.c b/test/api_test/odp_timer_ping.c
index d958815..fb640d8 100644
--- a/test/api_test/odp_timer_ping.c
+++ b/test/api_test/odp_timer_ping.c
@@ -322,9 +322,8 @@ int main(int argc __attribute__((__unused__)),
        ping_arg_t pingarg;
        odp_queue_t queue;
        odp_buffer_pool_t pool;
-       void *pool_base;
        int i;
-       odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
        if (odp_test_global_init() != 0)
                return -1;
@@ -337,14 +336,14 @@ int main(int argc __attribute__((__unused__)),
        /*
         * Create message pool
         */
-       shm = odp_shm_reserve("msg_pool",
-                             MSG_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
-       pool_base = odp_shm_addr(shm);
-
-       pool = odp_buffer_pool_create("msg_pool", pool_base, MSG_POOL_SIZE,
-                                     BUF_SIZE,
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_RAW);
+
+       params.buf_size  = BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = MSG_POOL_SIZE/BUF_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_RAW;
+
+       pool = odp_buffer_pool_create("msg_pool", ODP_SHM_NULL, &params);
+
        if (pool == ODP_BUFFER_POOL_INVALID) {
                LOG_ERR("Pool create failed.\n");
                return -1;
diff --git a/test/validation/odp_crypto.c b/test/validation/odp_crypto.c
index 03ca438..72cf0f0 100644
--- a/test/validation/odp_crypto.c
+++ b/test/validation/odp_crypto.c
@@ -25,26 +25,17 @@ CU_SuiteInfo odp_testsuites[] = {
 
 int tests_global_init(void)
 {
-       odp_shm_t shm;
-       void *pool_base;
+       odp_buffer_pool_param_t params;
        odp_buffer_pool_t pool;
        odp_queue_t out_queue;
 
-       shm = odp_shm_reserve("shm_packet_pool",
-                             SHM_PKT_POOL_SIZE,
-                             ODP_CACHE_LINE_SIZE, 0);
+       params.buf_size  = SHM_PKT_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_PKT_POOL_SIZE/SHM_PKT_POOL_BUF_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_PACKET;
 
-       pool_base = odp_shm_addr(shm);
-       if (!pool_base) {
-               fprintf(stderr, "Packet pool allocation failed.\n");
-               return -1;
-       }
+       pool = odp_buffer_pool_create("packet_pool", ODP_SHM_NULL, &params);
 
-       pool = odp_buffer_pool_create("packet_pool", pool_base,
-                                     SHM_PKT_POOL_SIZE,
-                                     SHM_PKT_POOL_BUF_SIZE,
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_PACKET);
        if (ODP_BUFFER_POOL_INVALID == pool) {
                fprintf(stderr, "Packet pool creation failed.\n");
                return -1;
@@ -55,20 +46,14 @@ int tests_global_init(void)
                fprintf(stderr, "Crypto outq creation failed.\n");
                return -1;
        }
-       shm = odp_shm_reserve("shm_compl_pool",
-                            SHM_COMPL_POOL_SIZE,
-                            ODP_CACHE_LINE_SIZE,
-                            ODP_SHM_SW_ONLY);
-       pool_base = odp_shm_addr(shm);
-       if (!pool_base) {
-               fprintf(stderr, "Completion pool allocation failed.\n");
-               return -1;
-       }
-       pool = odp_buffer_pool_create("compl_pool", pool_base,
-                                     SHM_COMPL_POOL_SIZE,
-                                     SHM_COMPL_POOL_BUF_SIZE,
-                                     ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_RAW);
+
+       params.buf_size  = SHM_COMPL_POOL_BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = SHM_COMPL_POOL_SIZE/SHM_COMPL_POOL_BUF_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_RAW;
+
+       pool = odp_buffer_pool_create("compl_pool", ODP_SHM_NULL, &params);
+
        if (ODP_BUFFER_POOL_INVALID == pool) {
                fprintf(stderr, "Completion pool creation failed.\n");
                return -1;
diff --git a/test/validation/odp_queue.c b/test/validation/odp_queue.c
index 9f42b6f..b5ccf9a 100644
--- a/test/validation/odp_queue.c
+++ b/test/validation/odp_queue.c
@@ -16,21 +16,14 @@ static int queue_contest = 0xff;
 static int init_queue_suite(void)
 {
        odp_buffer_pool_t pool;
-       void *pool_base;
-       odp_shm_t shm;
+       odp_buffer_pool_param_t params;
 
-       shm = odp_shm_reserve("msg_pool",
-                             MSG_POOL_SIZE, ODP_CACHE_LINE_SIZE, 0);
+       params.buf_size  = 0;
+       params.buf_align = ODP_CACHE_LINE_SIZE;
+       params.num_bufs  = 1024 * 10;
+       params.buf_type  = ODP_BUFFER_TYPE_RAW;
 
-       pool_base = odp_shm_addr(shm);
-
-       if (NULL == pool_base) {
-               printf("Shared memory reserve failed.\n");
-               return -1;
-       }
-
-       pool = odp_buffer_pool_create("msg_pool", pool_base, MSG_POOL_SIZE, 0,
-                                     ODP_CACHE_LINE_SIZE, ODP_BUFFER_TYPE_RAW);
+       pool = odp_buffer_pool_create("msg_pool", ODP_SHM_NULL, &params);
 
        if (ODP_BUFFER_POOL_INVALID == pool) {
                printf("Pool create failed.\n");
diff --git a/test/validation/odp_schedule.c b/test/validation/odp_schedule.c
index 9d410e4..31be742 100644
--- a/test/validation/odp_schedule.c
+++ b/test/validation/odp_schedule.c
@@ -510,22 +510,18 @@ static int create_queues(void)
 static int schd_suite_init(void)
 {
        odp_shm_t shm;
-       void *pool_base;
        odp_buffer_pool_t pool;
        test_globals_t *globals;
        thread_args_t *thr_args;
+       odp_buffer_pool_param_t params;
 
-       shm = odp_shm_reserve(SHM_MSG_POOL_NAME, MSG_POOL_SIZE,
-                             ODP_CACHE_LINE_SIZE, 0);
-       pool_base = odp_shm_addr(shm);
-       if (pool_base == NULL) {
-               printf("Shared memory reserve failed.\n");
-               return -1;
-       }
+       params.buf_size  = BUF_SIZE;
+       params.buf_align = 0;
+       params.num_bufs  = MSG_POOL_SIZE/BUF_SIZE;
+       params.buf_type  = ODP_BUFFER_TYPE_RAW;
+
+       pool = odp_buffer_pool_create(MSG_POOL_NAME, ODP_SHM_NULL, &params);
 
-       pool = odp_buffer_pool_create(MSG_POOL_NAME, pool_base, MSG_POOL_SIZE,
-                                     BUF_SIZE, ODP_CACHE_LINE_SIZE,
-                                     ODP_BUFFER_TYPE_RAW);
        if (pool == ODP_BUFFER_POOL_INVALID) {
                printf("Pool creation failed (msg).\n");
                return -1;
-- 
1.9.1


_______________________________________________
lng-odp mailing list
[email protected]
http://lists.linaro.org/mailman/listinfo/lng-odp

Reply via email to