diff --git a/.gitignore b/.gitignore
index dceaeaa1c4bd742c9511cf379e5fa73c17f5186d..f7cf6f582fc61e781eafa5efb51f3bebbf6a8859 100644
--- a/.gitignore
+++ b/.gitignore
@@ -136,3 +136,16 @@ libtool
 *.dSYM
 .DS_Store
 
+# Cmake 
+CMakeLists.txt.user
+CMakeCache.txt
+CMakeFiles
+CMakeScripts
+Testing
+Makefile
+cmake_install.cmake
+install_manifest.txt
+compile_commands.json
+CTestTestfile.cmake
+_deps
+CMakeUserPresets.json
diff --git a/CMakeLists.txt b/CMakeLists.txt
index 20739bda0487b03968b1c27166b5a89765b5b0cf..6d444fec2cdd41d95728af6e37117ddd9ac16b91 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -1,38 +1,95 @@
-cmake_minimum_required( VERSION 3.30 FATAL_ERROR )
+cmake_minimum_required( VERSION 3.27 FATAL_ERROR )
 
-project(libcdi VERSION 2.4.0 LANGUAGES C )
+message(VERBOSE "Entering libcdi")
+project(cdilib VERSION 2.5.1 LANGUAGES C )
 set(CMAKE_C_STANDARD 11)
 
+include(CTest)
 include(CheckIncludeFile)
 
-configure_file (
-  "${PROJECT_SOURCE_DIR}/cmake/cdi_config.h.in"
-  "${PROJECT_SOURCE_DIR}/src/config.h"
-  )
-
-find_package(hdf5 REQUIRED)
-if(hdf5_FOUND)
-    message(STATUS "HDF5 shaed found")
-else()
-    message(FATAL_ERROR "HDF shared not found")
-endif()
-
-#NetCDF
-find_package(netCDF COMPONENTS C REQUIRED)
-set(netcdf_flag HAVE_LIBNETCDF)
-set(netcdf_flag ${HAVE_LIBNETCDF})
-check_include_files("netcdf.h" ${netcdf_flag} C)
-if (${netCDF_FOUND})
-    message(STATUS "added compile definition HAVE_LIBNETCDF=${netCDF_FOUND}")
-    add_compile_definitions(HAVE_LIBNETCDF=${netCDF_FOUND})
-else()
-  message(WARNING "netcdf not found, compiling without netcdf")
-endif ()
+# Finding libraries
+### -------------- Pthread ---------------------------
+
+option(CDI_PTHREAD "Use the pthread library [default=ON]" ON)
+if(${CDI_PTHREAD})
+  include(FindThreads)
+  find_package(Threads REQUIRED)
+  set_target_properties(Threads::Threads PROPERTIES THREADS_PREFER_PTHREAD_FLAG TRUE)
+  list(APPEND cdi_compile_defs HAVE_PTHREAD=1)
+  list(APPEND cdi_linked_libs pthread)
+endif()
+
+# enable default internal libs
+option(CDI_LIBGRIB "GRIB support [default=ON]" ON)
+if(${CDI_LIBGRIB})
+  list(APPEND cdi_compile_defs HAVE_LIBGRIB=1)
+endif()
+
+option(CDI_LIBGRIBEX "Use the CGRIBEX library [default=ON]" ON)
+if(${CDI_LIBGRIBEX})
+  list(APPEND cdi_compile_defs LIBCGRIBEX=1)
+endif()
+
+option(CDI_EXTRA "Use the extra library [default=ON]" ON)
+if(${CDI_EXTRA})
+  list(APPEND cdi_compile_defs HAVE_LIBEXTRA=1)
+endif()
+
+option(CDI_IEG "Use the extra library [default=ON]" ON)
+if(${CDI_IEG})
+  list(APPEND cdi_compile_defs HAVE_LIBIEG=1)
+endif()
+
+option(CDI_SERVICE "Use the extra library [default=ON]" ON)
+if(${CDI_SERVICE})
+  list(APPEND cdi_compile_defs HAVE_LIBSERVICE=1)
+endif()
+
+# ecCodes
+option(CDI_ECCODES "Use the eccodes library [default=ON]" ON)
+if(${CDI_ECCODES} OR eccodes_ROOT)
+  find_package(eccodes)
+  if (${eccodes_FOUND})
+    list(APPEND cdi_compile_defs HAVE_LIBGRIB_API=${eccodes_FOUND})
+    message(VERBOSE "added compile definition HAVE_LIBGRIB_API=${eccodes_FOUND}")
+    list(APPEND cdi_linked_libs eccodes)
+  else()
+    message(WARNING "eccodes not found, compiling without eccodes")
+  endif ()
+endif()
 
+# NetCDF
+option(CDI_NETCDF "Use the netcdf library [default=ON]" ON)
+if(${CDI_NETCDF} OR netCDF_ROOT )
+  find_package(netCDF COMPONENTS C REQUIRED)
+  if (TARGET netCDF::netcdf)
+    list(APPEND cdi_compile_defs
+      HAVE_LIBNETCDF=${netCDF_FOUND}
+      HAVE_LIBNC_DAP=${netCDF_FOUND}
+      HAVE_NETCDF4=${netCDF_FOUND}
+      HAVE_LIBGRIB_API=${netCDF_FOUND}
+      HAVE_LIBGRIB=${netCDF_FOUND}
+    )
+    list(APPEND cdi_linked_libs netCDF::netcdf)
+  else()
+    message(WARNING "netcdf target not found, compiling without netcdf")
+  endif ()
+endif()
+
+message(VERBOSE "looking for config.h in: ${PROJECT_BINARY_DIR}/src")
 
 #adding subdirectories
 ## lib and general files
+list(APPEND cdi_compile_defs CDI=1 CDI_SIZE_TYPE=size_t PACKAGE_NAME="${PROJECT_NAME}" VERSION="${CMAKE_PROJECT_VERSION}")
 add_subdirectory(src)
-## cdi executable
-add_subdirectory(app)
 
+#app
+option(CDI_BUILD_APP "Build the app" ON)
+if (CDI_BUILD_APP)
+  add_subdirectory(app)
+endif ()
+
+#tests
+if (BUILD_TESTING)
+  add_subdirectory(tests)
+endif ()
diff --git a/ChangeLog b/ChangeLog
index 60d0a39cd565c2e01a08634b16d6a5fcfd979c00..b1e9272c0692a43be5c759aa014b62854d57147e 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,3 +1,45 @@
+2025-03-05  Uwe Schulzweida
+
+	* Version 2.5.1 released
+
+2025-02-28  Uwe Schulzweida
+
+	* taxis: added support for CDI_KEY_DATATYPE
+
+2025-02-12  Uwe Schulzweida
+
+	* GRIB_API: Handle LLAM as LCC
+
+2025-02-06  Uwe Schulzweida
+
+	* Added obsolete functions vlistNgrids() and vlistNzaxis() for ParaView vtkCDIReader
+
+2025-02-01  Uwe Schulzweida
+
+	* Added support for READ_CELL_CENTER=false
+
+2025-01-30  Uwe Schulzweida
+
+	* Added CDI_KEY_CHUNKSIZE_DIMX
+	* Added CDI_KEY_CHUNKSIZE_DIMY
+
+2025-01-29  Uwe Schulzweida
+
+	* Added CDI_KEY_CHUNKSIZE_DIMZ
+	* Added CDI_KEY_CHUNKSIZE_DIMT
+
+2025-01-14  Uwe Schulzweida
+
+	* NetCDF: Fix error in scanning coordinates attribute
+
+2024-12-28  Uwe Schulzweida
+
+	* NetCDF4: improved calculation of output chunk size
+
+2024-12-23  Uwe Schulzweida
+
+	* NetCDF4: improved calculation of input chunk cache size for numStep>1 and zSize>1
+
 2024-11-28  Uwe Schulzweida
 
         * using CGRIBEX library version 2.3.1
diff --git a/Makefile.am b/Makefile.am
index e6b9801be55fe18fe099d3ac1734f03f7f994bc1..6498136e51ab75b303d22893639d516302655adb 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -11,6 +11,7 @@ doc/cdi_fman.pdf:
 
 EXTRA_DIST = \
   LICENSE \
+  CMakeLists.txt \
   config/interface.rb \
   doc/cdi_cman.pdf \
   doc/cdi_fman.pdf \
diff --git a/NEWS b/NEWS
index d21b4942d545009488a8b484d6f3d109096da300..d05d2f19a2af320a4b8a230800d10148067b0809 100644
--- a/NEWS
+++ b/NEWS
@@ -1,6 +1,12 @@
 CDI NEWS
 --------
 
+Version 2.5.1 (05 Mar 2025):
+
+   Changes:
+     * NetCDF4: improved calculation of output chunk size
+     * NetCDF4: improved calculation of input chunk cache size for numStep>1 and zSize>1
+
 Version 2.5.0 (28 Nov 2024):
 
    Changes:
diff --git a/app/CMakeLists.txt b/app/CMakeLists.txt
index ce87f49683bff30ea6f52c4f06e33dc4d1bf135e..4264972e74952dfc37e3368f88c30b0ced8ecc6e 100644
--- a/app/CMakeLists.txt
+++ b/app/CMakeLists.txt
@@ -2,11 +2,11 @@ add_executable(cdi
   cdi.c printinfo.c
 )
 
-target_include_directories(cdi PUBLIC
+target_include_directories(cdi PRIVATE
   $<BUILD_INTERFACE:${PROJECT_SOURCE_DIR}/src>
   $<INSTALL_INTERFACE:include
 )
 
-target_link_libraries(cdi PUBLIC cdilib netCDF::netcdf ${HDF5_LIBS} pthread)
-
-target_compile_definitions(cdi PUBLIC HAVE_CONFIG_H)
+target_link_libraries(cdi PRIVATE cdilib ${cdi_linked_libs})
+target_include_directories(cdi PRIVATE "${PROJECT_BINARY_DIR}/src/config.h")
+target_compile_definitions(cdi PRIVATE  ${cdi_netcdf_definitions})
diff --git a/app/Makefile.am b/app/Makefile.am
index 421e8d9f5c74e39f0394204dbcf0f789c8166422..53f1384b97a89e746ef0e5ce46fcd9a042a623c1 100644
--- a/app/Makefile.am
+++ b/app/Makefile.am
@@ -2,6 +2,8 @@ bin_PROGRAMS    =
 noinst_PROGRAMS =
 check_PROGRAMS  =
 
+EXTRA_DIST = CMakeLists.txt
+
 if ENABLE_CDI_APP
 if ENABLE_CDI_LIB
 bin_PROGRAMS += cdi
diff --git a/app/printinfo.c b/app/printinfo.c
index ccbeea0049bdb477e3734ebe858d7251880cf61f..8fd902e0b769628db06bbcf9e152f9938f603bf2 100644
--- a/app/printinfo.c
+++ b/app/printinfo.c
@@ -300,8 +300,8 @@ printGridInfoKernel(int gridID, int index, bool lproj)
   size_t xsize = (size_t) gridInqXsize(gridID);
   size_t ysize = (size_t) gridInqYsize(gridID);
 
-  // int prec     = gridInqDatatype(gridID);
-  // int dig = (prec == CDI_DATATYPE_FLT64) ? 15 : 7;
+  // int datatype;
+  // cdiInqKeyInt(gridID, CDI_GLOBAL, CDI_KEY_DATATYPE, &datatype);
   int dig = 7;
 
   if (!lproj)
@@ -537,8 +537,9 @@ printZaxisInfo(int vlistID)
       int ltype = 0;
       cdiInqKeyInt(zaxisID, CDI_GLOBAL, CDI_KEY_TYPEOFFIRSTFIXEDSURFACE, &ltype);
       int levelsize = zaxisInqSize(zaxisID);
-      // int prec      = zaxisInqDatatype(zaxisID);
-      // int dig = (prec == CDI_DATATYPE_FLT64) ? 15 : 7;
+      // int datatype;
+      // cdiInqKeyInt(gridID, CDI_GLOBAL, CDI_KEY_DATATYPE, &datatype);
+      // int dig = (datatype == CDI_DATATYPE_FLT64) ? 15 : 7;
 
       zaxisName(zaxistype, zaxisname);
       int length = CDI_MAX_NAME;
diff --git a/cmake/cdi_config.h.in b/cmake/cdi_config.h.in
deleted file mode 100644
index e3bfd3225b75696f7fcb8e82af0f6f5b8acccba8..0000000000000000000000000000000000000000
--- a/cmake/cdi_config.h.in
+++ /dev/null
@@ -1,20 +0,0 @@
-#ifndef cdi_config_h
-#define cdi_config_h
-
-#define PACKAGE_NAME      "@PROJECT_NAME@"
-
-#define VERSION           "@PROJECT_VERSION@"
-
-#define CDI               1
-
-#define HAVE_LIBGRIB      1
-#define HAVE_LIBCGRIBEX   1
-#define HAVE_LIBEXTRA     1
-#define HAVE_LIBSERVICE   1
-#define HAVE_LIBIEG       1
-
-#define HAVE_NETCDF       @HAVE_NETCDF@
-#define HAVE_NETCDF_NC4   @netCDF_HAS_NC4@
-#define NETCDF_FOUND      @NetCDF_FOUND@
-
-#endif /* cdi_config_h */
diff --git a/configure.ac b/configure.ac
index caa057f45a4237c314588e343f073fcbedc14504..16a3b7051565555d6cf4eab1ca5603e2c4e0ecc4 100644
--- a/configure.ac
+++ b/configure.ac
@@ -7,7 +7,7 @@
 AC_PREREQ([2.69])
 LT_PREREQ([2.4.6])
 
-AC_INIT([cdi],[2.5.0],[https://mpimet.mpg.de/cdi])
+AC_INIT([cdi],[2.5.1],[https://mpimet.mpg.de/cdi])
 AC_DEFINE_UNQUOTED(CDI, ["$PACKAGE_VERSION"], [CDI version])
 
 AC_CONFIG_AUX_DIR([config])
@@ -21,6 +21,9 @@ AM_INIT_AUTOMAKE([1.16.1 foreign])
 AM_MAINTAINER_MODE([enable])
 AM_EXTRA_RECURSIVE_TARGETS([examples])
 
+AC_MSG_NOTICE([setting up libtool])
+LT_INIT
+
 dnl Tests for the C compiler:
 dnl   Make sure conftest.dSYM is removed when configured on MacOS with
 dnl   CFLAGS='-g':
diff --git a/examples/pio/compareResourcesArray.c b/examples/pio/compareResourcesArray.c
index 0082d60caee0271025039752166cee9fd0e94bd0..4eb771089b26c079bb3cbce5fa842bab1cc4cae1 100644
--- a/examples/pio/compareResourcesArray.c
+++ b/examples/pio/compareResourcesArray.c
@@ -60,7 +60,7 @@ defineGrid()
   cdiDefKeyString(gridID, CDI_XAXIS, CDI_KEY_UNITS, "myXunits");
   cdiDefKeyString(gridID, CDI_YAXIS, CDI_KEY_UNITS, "myYunits");
 
-  gridDefDatatype(gridID, DOUBLE_PRECISION);
+  cdiDefKeyInt(gridID, CDI_GLOBAL, CDI_KEY_DATATYPE, DOUBLE_PRECISION);
   gridDefTrunc(gridID, 1);
 
   cdiDefKeyInt(gridID, CDI_GLOBAL, CDI_KEY_NUMBEROFGRIDUSED, 6);
diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt
index 1f88fe09b84956160e7b52fb11ea0605f0b274a3..26ea963f1b0e27f9204c6cf8a2473d665a556bda 100644
--- a/src/CMakeLists.txt
+++ b/src/CMakeLists.txt
@@ -61,6 +61,7 @@ list( APPEND cdi_src_files
   grb_write.c
   gribapi.c
   gribapi.h
+  gribapi_utilities.c
   gribapi_utilities.h
   grid.c
   grid.h
@@ -103,6 +104,7 @@ list( APPEND cdi_src_files
   stream_ext.h
   stream_grb.c
   stream_grb.h
+  stream_gribapi.c
   stream_gribapi.h
   stream_ieg.c
   stream_ieg.h
@@ -137,12 +139,95 @@ list( APPEND cdi_src_files
   zaxis.c
   zaxis.h
 )
+list( APPEND cdi_pio_files
+  cdipio.h
+  pio.c
+  pio.h
+  pio_cdf_int.c
+  pio_cdf_int.h
+  pio_client.c
+  pio_client.h
+  pio_comm.c
+  pio_comm.h
+  pio_conf.c
+  pio_conf.h
+  pio_dbuffer.c
+  pio_dbuffer.h
+  pio_dist_grid.c
+  pio_dist_grid.h
+  pio_id_set.h
+  pio_idxlist_cache.c
+  pio_idxlist_cache.h
+  pio_impl.h
+  pio_interface.c
+  pio_interface.h
+  pio_mpi_fw_at_all.c
+  pio_mpi_fw_at_reblock.c
+  pio_mpi_fw_ordered.c
+  pio_mpinonb.c
+  pio_posixasynch.c
+  pio_posixfpguardsendrecv.c
+  pio_posixnonb.c
+  pio_record_send.c
+  pio_roles.c
+  pio_rpc.c
+  pio_rpc.h
+  pio_serialize.c
+  pio_serialize.h
+  pio_server.c
+  pio_server.h
+  pio_util.c
+  pio_util.h
+  pio_xmap_cache.c
+  pio_xmap_cache.h
+)
+
+list( APPEND cdi_unknown
+  cfortran.h
+  getline.c
+  make_fint.c
+  resource_unpack.c
+  resource_unpack.h
+)
+
+option(CDI_BUILD_UNKNOWN "Build unknown sources in libcdi" ON)
+mark_as_advanced(CDI_BUILD_UNKNOWN)
+if (CDI_BUILD_UNKNOWN)
+  list(APPEND cdi_src_files ${cdi_unknown})
+endif ()
+
+# Support exporting all symbolds on Windows
+set(CMAKE_WINDOWS_EXPORT_ALL_SYMBOLS ON)
 
 add_library(cdilib
   ${cdi_src_files}
 #  INSTALL_HEADERS_LIST  cdi.h
 )
-find_package(HDF5 REQUIRED COMPONENTS C REQUIRED)
 
-target_link_libraries(cdilib PUBLIC netCDF::netcdf ${HDF5_LIBS})
-target_compile_definitions(cdilib PUBLIC HAVE_CONFIG_H)
+target_include_directories(cdilib PRIVATE "${PROJECT_BINARY_DIR}/src/config.h")
+target_link_libraries(cdilib ${cdi_linked_libs})
+target_compile_definitions(cdilib PRIVATE PACKAGE_NAME="${PROJECT_NAME}" VERSION="${CMAKE_PROJECT_VERSION}" ${cdi_compile_defs})
+
+add_library(cdilib::cdilib ALIAS cdilib)
+
+include(GNUInstallDirs)
+install(FILES cdi.h calendar.h cdi_datetime.h julian_date.h DESTINATION "${CMAKE_INSTALL_INCLUDEDIR}")
+install(TARGETS cdilib
+  ARCHIVE DESTINATION "${CMAKE_INSTALL_LIBDIR}"
+  RUNTIME DESTINATION "${CMAKE_INSTALL_BINDIR}"
+  LIBRARY DESTINATION "${CMAKE_INSTALL_LIBDIR}")
+
+include(CMakePackageConfigHelpers)
+configure_package_config_file(
+  "${CMAKE_CURRENT_SOURCE_DIR}/cmake/cdi/cdi-config.cmake.in" "${CMAKE_BINARY_DIR}/cmake/cdi-config.cmake"
+  INSTALL_DESTINATION "${CMAKE_INSTALL_LIBDIR}/cmake/cdi")
+write_basic_package_version_file(
+  "${CMAKE_BINARY_DIR}/cmake/cdi-config-version.cmake"
+  VERSION "${PACKAGE_VERSION}"
+  COMPATIBILITY SameMajorVersion)
+install(
+  FILES
+    "${CMAKE_BINARY_DIR}/cmake/cdi-config.cmake"
+    "${CMAKE_BINARY_DIR}/cmake/cdi-config-version.cmake"
+  DESTINATION
+    "${CMAKE_INSTALL_LIBDIR}/cmake/cdi")
diff --git a/src/Makefile.am b/src/Makefile.am
index c38810106b8d353990dd954393643770db46d5e2..1903a6ac7ff5a03aeff61e16a1a5f3be9cba0bd1 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -3,6 +3,7 @@ check_LTLIBRARIES =
 noinst_LTLIBRARIES =
 EXTRA_PROGRAMS = make_fint
 EXTRA_DIST = \
+  CMakeLists.txt \
   cdilib.c \
   make_cdilib
 
diff --git a/src/cdf.c b/src/cdf.c
index 78e7fff9ff3579a3718212be6e11917fdf186367..43b1df25f837623cd7e59141b7461a30894c7a97 100644
--- a/src/cdf.c
+++ b/src/cdf.c
@@ -200,7 +200,7 @@ static void
 cdfCloseFile(int fileID)
 {
 
-  if (CDF_Debug) Message( "Closing cdf file: %d", fileID);
+  if (CDF_Debug) Message("Closing cdf file: %d", fileID);
   cdf_close(fileID);
 }
 
@@ -209,6 +209,7 @@ cdfClose(int fileID)
 {
   cdfCloseFile(fileID);
 }
+
 #endif
 /*
  * Local Variables:
diff --git a/src/cdf_int.c b/src/cdf_int.c
index c7c03a5ea7b949de077d50ce914ccb8cd2b874e4..1c0eff0b29acbea41368d4b24676db4f5f1c4f6d 100644
--- a/src/cdf_int.c
+++ b/src/cdf_int.c
@@ -32,7 +32,6 @@ cdf__create(const char *path, int cmode, int *ncidp)
 {
   int status = -1;
   size_t chunksizehint = 0;
-
   size_t initialsz = 0;
 
 #if defined(__SX__) || defined(ES)
@@ -128,14 +127,14 @@ cdf_enddef(int ncid, int streamID)
 }
 
 void
-cdf__enddef(int ncid, int streamID, const size_t hdr_pad)
+cdf__enddef(int ncid, int streamID, size_t hdr_pad)
 {
-  const size_t v_align = 4UL;    // [B] Alignment of beginning of data section for fixed variables
-  const size_t v_minfree = 0UL;  // [B] Pad at end of data section for fixed size variables
-  const size_t r_align = 4UL;    // [B] Alignment of beginning of data section for record variables
+  size_t v_align = 4UL;    // [B] Alignment of beginning of data section for fixed variables
+  size_t v_minfree = 0UL;  // [B] Pad at end of data section for fixed size variables
+  size_t r_align = 4UL;    // [B] Alignment of beginning of data section for record variables
 
   // nc_enddef(ncid) is equivalent to nc__enddef(ncid, 0, 4, 0, 4)
-  cdi_nc__enddef_funcp my_nc__enddef = (cdi_nc__enddef_funcp) namespaceSwitchGet(NSSWITCH_NC_ENDDEF).func;
+  cdi_nc__enddef_funcp my_nc__enddef = (cdi_nc__enddef_funcp) namespaceSwitchGet(NSSWITCH_NC__ENDDEF).func;
   int status = my_nc__enddef(ncid, streamID, hdr_pad, v_align, v_minfree, r_align);
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
@@ -151,7 +150,6 @@ void
 cdf_inq(int ncid, int *ndimsp, int *nvarsp, int *ngattsp, int *unlimdimidp)
 {
   int status = nc_inq(ncid, ndimsp, nvarsp, ngattsp, unlimdimidp);
-
   if (CDF_Debug || status != NC_NOERR)
     Message("ncid=%d  ndims=%d  nvars=%d  ngatts=%d  unlimid=%d", ncid, *ndimsp, *nvarsp, *ngattsp, *unlimdimidp);
 
@@ -162,9 +160,7 @@ void
 cdf_def_dim(int ncid, const char *name, size_t len, int *dimidp)
 {
   int status = nc_def_dim(ncid, name, len, dimidp);
-
-  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  name=%s  len=%d", ncid, name, len);
-
+  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  name=%s  len=%zu", ncid, name, len);
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -172,9 +168,7 @@ void
 cdf_inq_dimid(int ncid, const char *name, int *dimidp)
 {
   int status = nc_inq_dimid(ncid, name, dimidp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  name=%s  dimid=%d", ncid, name, *dimidp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -182,9 +176,7 @@ void
 cdf_inq_dim(int ncid, int dimid, char *name, size_t *lengthp)
 {
   int status = nc_inq_dim(ncid, dimid, name, lengthp);
-
-  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  dimid=%d  length=%d  name=%s", ncid, dimid, *lengthp, name);
-
+  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  dimid=%d  length=%zu  name=%s", ncid, dimid, *lengthp, name);
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -192,9 +184,7 @@ void
 cdf_inq_dimname(int ncid, int dimid, char *name)
 {
   int status = nc_inq_dimname(ncid, dimid, name);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  dimid=%d  name=%s", ncid, dimid, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -202,9 +192,7 @@ void
 cdf_inq_dimlen(int ncid, int dimid, size_t *lengthp)
 {
   int status = nc_inq_dimlen(ncid, dimid, lengthp);
-
-  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  dimid=%d  length=%d", ncid, dimid, *lengthp);
-
+  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  dimid=%d  length=%zu", ncid, dimid, *lengthp);
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -234,9 +222,7 @@ void
 cdf_inq_varid(int ncid, const char *name, int *varidp)
 {
   int status = nc_inq_varid(ncid, name, varidp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  name=%s  varid=%d", ncid, name, *varidp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -244,9 +230,7 @@ void
 cdf_inq_nvars(int ncid, int *nvarsp)
 {
   int status = nc_inq_nvars(ncid, nvarsp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  nvars=%d", ncid, *nvarsp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -254,7 +238,6 @@ void
 cdf_inq_var(int ncid, int varid, char *name, nc_type *xtypep, int *ndimsp, int dimids[], int *nattsp)
 {
   int status = nc_inq_var(ncid, varid, name, xtypep, ndimsp, dimids, nattsp);
-
   if (CDF_Debug || status != NC_NOERR)
     Message("ncid=%d  varid=%d  ndims=%d  xtype=%d  natts=%d  name=%s", ncid, varid, *ndimsp, *xtypep, *nattsp, name);
 
@@ -265,9 +248,7 @@ void
 cdf_inq_varname(int ncid, int varid, char *name)
 {
   int status = nc_inq_varname(ncid, varid, name);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  name=%s", ncid, varid, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -275,9 +256,7 @@ void
 cdf_inq_vartype(int ncid, int varid, nc_type *xtypep)
 {
   int status = nc_inq_vartype(ncid, varid, xtypep);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  xtype=%s", ncid, varid, *xtypep);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -285,9 +264,7 @@ void
 cdf_inq_varndims(int ncid, int varid, int *ndimsp)
 {
   int status = nc_inq_varndims(ncid, varid, ndimsp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -295,9 +272,7 @@ void
 cdf_inq_vardimid(int ncid, int varid, int dimids[])
 {
   int status = nc_inq_vardimid(ncid, varid, dimids);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -305,9 +280,7 @@ void
 cdf_inq_varnatts(int ncid, int varid, int *nattsp)
 {
   int status = nc_inq_varnatts(ncid, varid, nattsp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  nattsp=%d", ncid, varid, *nattsp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -315,9 +288,7 @@ void
 cdf_put_var_text(int ncid, int varid, const char *tp)
 {
   int status = nc_put_var_text(ncid, varid, tp);
-
   if (CDF_Debug || status != NC_NOERR) Message("%d %d %s", ncid, varid, tp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -325,9 +296,7 @@ void
 cdf_put_var_short(int ncid, int varid, const short *sp)
 {
   int status = nc_put_var_short(ncid, varid, sp);
-
   if (CDF_Debug || status != NC_NOERR) Message("%d %d %hd", ncid, varid, *sp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -335,9 +304,7 @@ void
 cdf_put_var_int(int ncid, int varid, const int *ip)
 {
   int status = nc_put_var_int(ncid, varid, ip);
-
   if (CDF_Debug || status != NC_NOERR) Message("%d %d %d", ncid, varid, *ip);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -345,9 +312,7 @@ void
 cdf_put_var_long(int ncid, int varid, const long *lp)
 {
   int status = nc_put_var_long(ncid, varid, lp);
-
   if (CDF_Debug || status != NC_NOERR) Message("%d %d %ld", ncid, varid, *lp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -355,9 +320,7 @@ void
 cdf_put_var_float(int ncid, int varid, const float *fp)
 {
   int status = nc_put_var_float(ncid, varid, fp);
-
   if (CDF_Debug || status != NC_NOERR) Message("%d %d %f", ncid, varid, *fp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -420,7 +383,6 @@ void
 cdf_put_vara_double(int ncid, int varid, const size_t start[], const size_t count[], const double *dp)
 {
   int status = nc_put_vara_double(ncid, varid, start, count, dp);
-
   if (CDF_Debug || status != NC_NOERR)
     {
       char name[256];
@@ -443,7 +405,6 @@ void
 cdf_put_vara_float(int ncid, int varid, const size_t start[], const size_t count[], const float *fp)
 {
   int status = nc_put_vara_float(ncid, varid, start, count, fp);
-
   if (CDF_Debug || status != NC_NOERR)
     {
       char name[256];
@@ -466,9 +427,7 @@ void
 cdf_put_vara(int ncid, int varid, const size_t start[], const size_t count[], const void *cp)
 {
   int status = nc_put_vara(ncid, varid, start, count, cp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -476,9 +435,7 @@ void
 cdf_get_vara(int ncid, int varid, const size_t start[], const size_t count[], void *cp)
 {
   int status = nc_get_vara(ncid, varid, start, count, cp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -486,9 +443,7 @@ void
 cdf_get_vara_int(int ncid, int varid, const size_t start[], const size_t count[], int *dp)
 {
   int status = nc_get_vara_int(ncid, varid, start, count, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -496,9 +451,7 @@ void
 cdf_get_vara_double(int ncid, int varid, const size_t start[], const size_t count[], double *dp)
 {
   int status = nc_get_vara_double(ncid, varid, start, count, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  start[0]=%zu  count[0]=%zu", ncid, varid, start[0], count[0]);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -506,9 +459,7 @@ void
 cdf_get_vara_float(int ncid, int varid, const size_t start[], const size_t count[], float *fp)
 {
   int status = nc_get_vara_float(ncid, varid, start, count, fp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  start[0]=%zu  count[0]=%zu", ncid, varid, start[0], count[0]);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -516,9 +467,7 @@ void
 cdf_get_vara_text(int ncid, int varid, const size_t start[], const size_t count[], char *tp)
 {
   int status = nc_get_vara_text(ncid, varid, start, count, tp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -526,9 +475,7 @@ void
 cdf_get_vara_uchar(int ncid, int varid, const size_t start[], const size_t count[], unsigned char *tp)
 {
   int status = nc_get_vara_uchar(ncid, varid, start, count, tp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -536,9 +483,7 @@ void
 cdf_put_var_double(int ncid, int varid, const double *dp)
 {
   int status = nc_put_var_double(ncid, varid, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  val0=%f", ncid, varid, *dp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -546,9 +491,7 @@ void
 cdf_get_var1_text(int ncid, int varid, const size_t index[], char *tp)
 {
   int status = nc_get_var1_text(ncid, varid, index, tp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -556,9 +499,7 @@ void
 cdf_get_var1_double(int ncid, int varid, const size_t index[], double *dp)
 {
   int status = nc_get_var1_double(ncid, varid, index, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -566,9 +507,7 @@ void
 cdf_put_var1_double(int ncid, int varid, const size_t index[], const double *dp)
 {
   int status = nc_put_var1_double(ncid, varid, index, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  val=%f", ncid, varid, *dp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -576,9 +515,7 @@ void
 cdf_get_var_text(int ncid, int varid, char *tp)
 {
   int status = nc_get_var_text(ncid, varid, tp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -586,9 +523,7 @@ void
 cdf_get_var_short(int ncid, int varid, short *sp)
 {
   int status = nc_get_var_short(ncid, varid, sp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -596,9 +531,7 @@ void
 cdf_get_var_int(int ncid, int varid, int *ip)
 {
   int status = nc_get_var_int(ncid, varid, ip);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -606,9 +539,7 @@ void
 cdf_get_var_long(int ncid, int varid, long *lp)
 {
   int status = nc_get_var_long(ncid, varid, lp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -616,9 +547,7 @@ void
 cdf_get_var_float(int ncid, int varid, float *fp)
 {
   int status = nc_get_var_float(ncid, varid, fp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d", ncid, varid);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -626,9 +555,7 @@ void
 cdf_get_var_double(int ncid, int varid, double *dp)
 {
   int status = nc_get_var_double(ncid, varid, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  val[0]=%f", ncid, varid, *dp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -636,9 +563,7 @@ void
 cdf_copy_att(int ncid_in, int varid_in, const char *name, int ncid_out, int varid_out)
 {
   int status = nc_copy_att(ncid_in, varid_in, name, ncid_out, varid_out);
-
   if (CDF_Debug || status != NC_NOERR) Message("%d %d %s %d %d", ncid_in, varid_out, name, ncid_out, varid_out);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -646,9 +571,7 @@ void
 cdf_put_att_text(int ncid, int varid, const char *name, size_t len, const char *tp)
 {
   int status = nc_put_att_text(ncid, varid, name, len, tp);
-
-  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  text=%.*s", ncid, varid, name, (int) len, tp);
-
+  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  text='%.*s'", ncid, varid, name, (int) len, tp);
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -656,11 +579,8 @@ void
 cdf_put_att_int(int ncid, int varid, const char *name, nc_type xtype, size_t len, const int *ip)
 {
   int status = nc_put_att_int(ncid, varid, name, xtype, len, ip);
-
   if (status == NC_ERANGE) status = NC_NOERR;
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  val=%d", ncid, varid, name, *ip);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -668,9 +588,7 @@ void
 cdf_put_att_float(int ncid, int varid, const char *name, nc_type xtype, size_t len, const float *dp)
 {
   int status = nc_put_att_float(ncid, varid, name, xtype, len, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  val=%g", ncid, varid, name, *dp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -678,9 +596,7 @@ void
 cdf_put_att_double(int ncid, int varid, const char *name, nc_type xtype, size_t len, const double *dp)
 {
   int status = nc_put_att_double(ncid, varid, name, xtype, len, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  val=%g", ncid, varid, name, *dp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -688,9 +604,7 @@ void
 cdf_get_att_text(int ncid, int varid, const char *name, char *tp)
 {
   int status = nc_get_att_text(ncid, varid, name, tp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  name=%s", ncid, varid, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -699,9 +613,7 @@ cdf_get_att_string(int ncid, int varid, const char *name, char **tp)
 {
 #ifdef HAVE_NETCDF4
   int status = nc_get_att_string(ncid, varid, name, tp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  name=%s", ncid, varid, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 #endif
 }
@@ -710,11 +622,8 @@ void
 cdf_get_att_int(int ncid, int varid, const char *name, int *ip)
 {
   int status = nc_get_att_int(ncid, varid, name, ip);
-
   if (status == NC_ERANGE) status = NC_NOERR;
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  val=%d", ncid, varid, name, *ip);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -723,11 +632,8 @@ cdf_get_att_longlong(int ncid, int varid, const char *name, long long *llp)
 {
 #ifdef HAVE_NETCDF4
   int status = nc_get_att_longlong(ncid, varid, name, llp);
-
   if (status == NC_ERANGE) status = NC_NOERR;
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  val=%lld", ncid, varid, name, *llp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 #endif
 }
@@ -736,9 +642,7 @@ void
 cdf_get_att_double(int ncid, int varid, const char *name, double *dp)
 {
   int status = nc_get_att_double(ncid, varid, name, dp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  val=%.9g", ncid, varid, name, *dp);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -746,9 +650,7 @@ void
 cdf_inq_att(int ncid, int varid, const char *name, nc_type *xtypep, size_t *lenp)
 {
   int status = nc_inq_att(ncid, varid, name, xtypep, lenp);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s", ncid, varid, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -756,9 +658,7 @@ void
 cdf_inq_atttype(int ncid, int varid, const char *name, nc_type *xtypep)
 {
   int status = nc_inq_atttype(ncid, varid, name, xtypep);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s", ncid, varid, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -766,9 +666,7 @@ void
 cdf_inq_attlen(int ncid, int varid, const char *name, size_t *lenp)
 {
   int status = nc_inq_attlen(ncid, varid, name, lenp);
-
-  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  len=%d", ncid, varid, name, *lenp);
-
+  if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s  len=%zu", ncid, varid, name, *lenp);
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -776,9 +674,7 @@ void
 cdf_inq_attname(int ncid, int varid, int attnum, char *name)
 {
   int status = nc_inq_attname(ncid, varid, attnum, name);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  attnum=%d  att=%s", ncid, varid, attnum, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -786,9 +682,7 @@ void
 cdf_inq_attid(int ncid, int varid, const char *name, int *attnump)
 {
   int status = nc_inq_attid(ncid, varid, name, attnump);
-
   if (CDF_Debug || status != NC_NOERR) Message("ncid=%d  varid=%d  att=%s", ncid, varid, name);
-
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 
@@ -797,10 +691,34 @@ void
 cdf_def_var_chunking(int ncid, int varid, int storage, const size_t *chunksizesp)
 {
   int status = nc_def_var_chunking(ncid, varid, storage, chunksizesp);
+  if (CDF_Debug || status != NC_NOERR)
+    Message("chunks=%zu/%zu/%zu/%zu", chunksizesp[0], chunksizesp[1], chunksizesp[2], chunksizesp[3]);
   if (status != NC_NOERR) Error("%s", nc_strerror(status));
 }
 #endif
 
+size_t
+cdf_xtype_to_numbytes(nc_type xtype)
+{
+  size_t numBytes = 8;
+
+  // clang-format off
+  if      (xtype == NC_BYTE  )  numBytes = 1;
+  else if (xtype == NC_CHAR  )  numBytes = 1;
+  else if (xtype == NC_SHORT )  numBytes = 2;
+  else if (xtype == NC_INT   )  numBytes = 4;
+  else if (xtype == NC_FLOAT )  numBytes = 4;
+#ifdef HAVE_NETCDF4
+  else if (xtype == NC_UBYTE )  numBytes = 1;
+  else if (xtype == NC_USHORT)  numBytes = 2;
+  else if (xtype == NC_LONG  )  numBytes = 4;
+  else if (xtype == NC_UINT  )  numBytes = 4;
+#endif
+  // clang-format on
+
+  return numBytes;
+}
+
 #endif
 /*
  * Local Variables:
diff --git a/src/cdf_int.h b/src/cdf_int.h
index 7a4931a49b5342f8dee4b5d41bb84afd32ef1beb..6afae60303733513b5e0f92ba88730428d323a86 100644
--- a/src/cdf_int.h
+++ b/src/cdf_int.h
@@ -13,7 +13,7 @@ void cdf_close(int ncid);
 
 void cdf_redef(int ncid);
 void cdf_enddef(int ncid, int streamID);
-void cdf__enddef(int ncid, int streamID, const size_t hdr_pad);
+void cdf__enddef(int ncid, int streamID, size_t hdr_pad);
 void cdf_sync(int ncid);
 
 void cdf_inq(int ncid, int *ndimsp, int *nvarsp, int *ngattsp, int *unlimdimidp);
@@ -99,6 +99,9 @@ int cdi_nc_enddef_serial(int ncid, int streamID);
 int cdi_nc__enddef_serial(int ncid, int streamID, size_t hdr_pad, size_t v_align, size_t v_minfree, size_t r_align);
 typedef int (*cdi_nc_enddef_funcp)(int ncid, int streamID);
 typedef int (*cdi_nc__enddef_funcp)(int ncid, int streamID, size_t hdr_pad, size_t v_align, size_t v_minfree, size_t r_align);
+
+size_t cdf_xtype_to_numbytes(nc_type xtype);
+
 #endif
 
 #endif /* CDF_INT_H */
diff --git a/src/cdf_lazy_grid.c b/src/cdf_lazy_grid.c
index af1315bddd49e2a1735dc04af15e27df67d50ecc..65339aec2935a81ef74454466783376f90ac989d 100644
--- a/src/cdf_lazy_grid.c
+++ b/src/cdf_lazy_grid.c
@@ -3,7 +3,9 @@
 #endif
 
 #ifdef HAVE_LIBNETCDF
+#include "dmemory.h"
 #include "stream_cdf.h"
+#include "cdf_int.h"
 #include "cdf_lazy_grid.h"
 
 static struct gridVirtTable cdfLazyGridVtable;
@@ -42,7 +44,7 @@ static void
 cdfLazyGridDelete(grid_t *grid)
 {
   struct cdfLazyGrid *cdfGrid = (struct cdfLazyGrid *) grid;
-  void (*baseDestroy)(grid_t * grid) = cdfGrid->baseVtable->destroy;
+  void (*baseDestroy)(grid_t *grid) = cdfGrid->baseVtable->destroy;
   cdfLazyGridDestroy(cdfGrid);
   baseDestroy(grid);
 }
@@ -194,7 +196,7 @@ cdfLazyGridInqXVal(grid_t *grid, SizeType index)
 {
   struct cdfLazyGrid *lazyGrid = (struct cdfLazyGrid *) grid;
   lock_lazy_load(lazyGrid);
-  const double rv = cdfLazyGridInqXYVal(grid, (size_t) index, &lazyGrid->xValsGet, grid->x.vals, grid->vtable->inqXValsPtr);
+  double rv = cdfLazyGridInqXYVal(grid, (size_t) index, &lazyGrid->xValsGet, grid->x.vals, grid->vtable->inqXValsPtr);
   unlock_lazy_load(lazyGrid);
   return rv;
 }
@@ -204,7 +206,7 @@ cdfLazyGridInqYVal(grid_t *grid, SizeType index)
 {
   struct cdfLazyGrid *lazyGrid = (struct cdfLazyGrid *) grid;
   lock_lazy_load(lazyGrid);
-  const double rv = cdfLazyGridInqXYVal(grid, (size_t) index, &lazyGrid->yValsGet, grid->y.vals, grid->vtable->inqYValsPtr);
+  double rv = cdfLazyGridInqXYVal(grid, (size_t) index, &lazyGrid->yValsGet, grid->y.vals, grid->vtable->inqYValsPtr);
   unlock_lazy_load(lazyGrid);
   return rv;
 }
@@ -315,10 +317,10 @@ cdfLazyGridCopyScalarFields(grid_t *gridptrOrig, grid_t *gridptrDup)
 static void
 cdfLazyGridCopyArrayFields(grid_t *gridptrOrig, grid_t *gridptrDup)
 {
-  const size_t reducedPointsSize = (size_t) gridptrOrig->reducedPointsSize;
-  const size_t gridsize = gridptrOrig->size;
-  const int gridtype = gridptrOrig->type;
-  const int irregular = (gridtype == GRID_CURVILINEAR || gridtype == GRID_UNSTRUCTURED);
+  size_t reducedPointsSize = (size_t) gridptrOrig->reducedPointsSize;
+  size_t gridsize = gridptrOrig->size;
+  int gridtype = gridptrOrig->type;
+  int irregular = (gridtype == GRID_CURVILINEAR || gridtype == GRID_UNSTRUCTURED);
 
   if (reducedPointsSize)
     {
@@ -328,28 +330,28 @@ cdfLazyGridCopyArrayFields(grid_t *gridptrOrig, grid_t *gridptrDup)
 
   if (gridptrOrig->x.vals != NULL && gridptrOrig->x.vals != cdfPendingLoad)
     {
-      const size_t size = irregular ? gridsize : gridptrOrig->x.size;
+      size_t size = irregular ? gridsize : gridptrOrig->x.size;
       gridptrDup->x.vals = (double *) Malloc(size * sizeof(double));
       memcpy(gridptrDup->x.vals, gridptrOrig->x.vals, size * sizeof(double));
     }
 
   if (gridptrOrig->y.vals != NULL && gridptrOrig->y.vals != cdfPendingLoad)
     {
-      const size_t size = irregular ? gridsize : gridptrOrig->y.size;
+      size_t size = irregular ? gridsize : gridptrOrig->y.size;
       gridptrDup->y.vals = (double *) Malloc(size * sizeof(double));
       memcpy(gridptrDup->y.vals, gridptrOrig->y.vals, size * sizeof(double));
     }
 
   if (gridptrOrig->x.bounds != NULL && gridptrOrig->x.bounds != cdfPendingLoad)
     {
-      const size_t size = (irregular ? gridsize : gridptrOrig->x.size) * (size_t) gridptrOrig->nvertex;
+      size_t size = (irregular ? gridsize : gridptrOrig->x.size) * (size_t) gridptrOrig->nvertex;
       gridptrDup->x.bounds = (double *) Malloc(size * sizeof(double));
       memcpy(gridptrDup->x.bounds, gridptrOrig->x.bounds, size * sizeof(double));
     }
 
   if (gridptrOrig->y.bounds != NULL && gridptrOrig->y.bounds != cdfPendingLoad)
     {
-      const size_t size = (irregular ? gridsize : gridptrOrig->y.size) * (size_t) gridptrOrig->nvertex;
+      size_t size = (irregular ? gridsize : gridptrOrig->y.size) * (size_t) gridptrOrig->nvertex;
       gridptrDup->y.bounds = (double *) Malloc(size * sizeof(double));
       memcpy(gridptrDup->y.bounds, gridptrOrig->y.bounds, size * sizeof(double));
     }
@@ -357,7 +359,7 @@ cdfLazyGridCopyArrayFields(grid_t *gridptrOrig, grid_t *gridptrDup)
   {
     if (gridptrOrig->area != NULL && gridptrOrig->area != cdfPendingLoad)
       {
-        const size_t size = gridsize;
+        size_t size = gridsize;
         gridptrDup->area = (double *) Malloc(size * sizeof(double));
         memcpy(gridptrDup->area, gridptrOrig->area, size * sizeof(double));
       }
@@ -365,14 +367,14 @@ cdfLazyGridCopyArrayFields(grid_t *gridptrOrig, grid_t *gridptrDup)
 
   if (gridptrOrig->mask != NULL)
     {
-      const size_t size = gridsize;
+      size_t size = gridsize;
       gridptrDup->mask = (mask_t *) Malloc(size * sizeof(mask_t));
       memcpy(gridptrDup->mask, gridptrOrig->mask, size * sizeof(mask_t));
     }
 
   if (gridptrOrig->mask_gme != NULL)
     {
-      const size_t size = gridsize;
+      size_t size = gridsize;
       gridptrDup->mask_gme = (mask_t *) Malloc(size * sizeof(mask_t));
       memcpy(gridptrDup->mask_gme, gridptrOrig->mask_gme, size * sizeof(mask_t));
     }
diff --git a/src/cdf_lazy_grid.h b/src/cdf_lazy_grid.h
index ac0184ffd2854bb33c15571e67b02989ed7d7074..3788e2622c534999a991e8b302c4254b1f7143a6 100644
--- a/src/cdf_lazy_grid.h
+++ b/src/cdf_lazy_grid.h
@@ -19,8 +19,6 @@
 
 #include <string.h>
 
-#include "dmemory.h"
-#include "cdf_int.h"
 #include "grid.h"
 
 struct xyValGet
diff --git a/src/cdf_read.c b/src/cdf_read.c
index 05622e06081bc7bfa5e099e941303d242be8c217..487ec7fad907651fbbac8fb578e3c08e341c641e 100644
--- a/src/cdf_read.c
+++ b/src/cdf_read.c
@@ -15,7 +15,6 @@
 #include "stream_cdf.h"
 #include "cdf_int.h"
 #include "vlist.h"
-#include "vlist_var.h"
 
 static void
 cdfReadGridTraj(stream_t *streamptr, int gridID)
@@ -24,8 +23,9 @@ cdfReadGridTraj(stream_t *streamptr, int gridID)
   int fileID = streamptr->fileID;
 
   int gridindex = vlistGridIndex(vlistID, gridID);
-  int ncLonId = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_X];
-  int ncLatId = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_Y];
+  const CdfGrid *cdfGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
+  int ncLonId = cdfGrid->ncIdVec[CDF_VARID_X];
+  int ncLatId = cdfGrid->ncIdVec[CDF_VARID_Y];
 
   int tsID = streamptr->curTsID;
   size_t ncStepIndex = (size_t) streamptr->tsteps[tsID].ncStepIndex;
@@ -47,6 +47,7 @@ cdfGetSlapDescription(stream_t *streamptr, int varID, size_t (*start)[MAX_DIMENS
   int zaxisID = vlistInqVarZaxis(vlistID, varID);
   int timetype = vlistInqVarTimetype(vlistID, varID);
   int gridindex = vlistGridIndex(vlistID, gridID);
+  const CdfGrid *cdfGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
   size_t ncStepIndex = (size_t) streamptr->tsteps[tsID].ncStepIndex;
 
   int xid = CDI_UNDEFID, yid = CDI_UNDEFID;
@@ -56,11 +57,11 @@ cdfGetSlapDescription(stream_t *streamptr, int varID, size_t (*start)[MAX_DIMENS
     }
   else
     {
-      xid = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_X];
-      yid = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_Y];
+      xid = cdfGrid->ncIdVec[CDF_DIMID_X];
+      yid = cdfGrid->ncIdVec[CDF_DIMID_Y];
     }
   int zaxisindex = vlistZaxisIndex(vlistID, zaxisID);
-  int zid = streamptr->zaxisID[zaxisindex];
+  int zid = streamptr->cdfInfo.zaxisIdVec[zaxisindex];
 
   int ndims = 0;
 #define addDimension(startCoord, length) \
@@ -376,7 +377,7 @@ cdf_inq_dimIds(stream_t *streamptr, int varId, int (*outDimIds)[4])
   int vlistID = streamptr->vlistID;
   int gridId = vlistInqVarGrid(vlistID, varId);
   int gridindex = vlistGridIndex(vlistID, gridId);
-  const int *ncIDs = streamptr->ncgrid[gridindex].ncIDs;
+  const int *ncIDs = streamptr->cdfInfo.cdfGridVec[gridindex].ncIdVec;
 
   switch (gridInqType(gridId))
     {
@@ -395,15 +396,16 @@ cdf_inq_dimIds(stream_t *streamptr, int varId, int (*outDimIds)[4])
 
   int zaxisID = vlistInqVarZaxis(vlistID, varId);
   int zaxisindex = vlistZaxisIndex(vlistID, zaxisID);
-  (*outDimIds)[2] = streamptr->zaxisID[zaxisindex];
+  (*outDimIds)[2] = streamptr->cdfInfo.zaxisIdVec[zaxisindex];
 }
 
 static size_t
 stream_inq_dimlen(stream_t *streamptr, int dimid)
 {
-  int ndims = streamptr->ncNumDims;
-  int *ncdimid = streamptr->ncDimID;
-  size_t *ncdimlen = streamptr->ncDimLen;
+  const CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  int ndims = cdfInfo->ncNumDims;
+  const int *ncdimid = cdfInfo->ncDimIdVec;
+  const size_t *ncdimlen = cdfInfo->ncDimLenVec;
   for (int i = 0; i < ndims; ++i)
     {
       if (dimid == ncdimid[i]) return ncdimlen[i];
@@ -490,8 +492,8 @@ cdfGetSliceSlapDescription(stream_t *streamptr, long tsID, int varID, int levelI
   if (skipdim == 1) addDimension(0, 1);
 
   int gridindex = vlistGridIndex(vlistID, gridId);
-  const ncgrid_t *ncGrid = &streamptr->ncgrid[gridindex];
-  bool readPart = (ncGrid->gridID == gridId && ncGrid->start != -1 && ncGrid->count != -1);
+  const CdfGrid *cdfGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
+  bool readPart = (cdfGrid->gridID == gridId && cdfGrid->start != -1 && cdfGrid->count != -1);
 
   for (int id = 0; id < 4; ++id)
     {
@@ -503,7 +505,7 @@ cdfGetSliceSlapDescription(stream_t *streamptr, long tsID, int varID, int levelI
         case 2:
         case 4:
           if (readPart && curDimId == dimIds[0])
-            addDimension((size_t) ncGrid->start, (size_t) ncGrid->count);
+            addDimension((size_t) cdfGrid->start, (size_t) cdfGrid->count);
           else
             addDimension(0, stream_inq_dimlen(streamptr, curDimId));
           break;
@@ -794,14 +796,13 @@ static size_t
 cdf_read_data(stream_t *streamptr, int recID, int memtype, void *data)
 {
   int tsID = streamptr->curTsID;
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
-  int levelID = streamptr->tsteps[tsID].records[recID].levelID;
+  recinfo_t *recInfo = &(streamptr->tsteps[tsID].recinfo[recID]);
 
   size_t numMissVals = 0;
   if (memtype == MEMTYPE_DOUBLE)
-    cdf_read_var_slice_DP(streamptr, tsID, varID, levelID, (double *) data, &numMissVals);
+    cdf_read_var_slice_DP(streamptr, tsID, recInfo->varID, recInfo->levelID, (double *) data, &numMissVals);
   else
-    cdf_read_var_slice_SP(streamptr, tsID, varID, levelID, (float *) data, &numMissVals);
+    cdf_read_var_slice_SP(streamptr, tsID, recInfo->varID, recInfo->levelID, (float *) data, &numMissVals);
 
   return numMissVals;
 }
@@ -815,16 +816,15 @@ typedef struct JobDescriptorCDF
 static JobArgsCDF
 job_args_init(stream_t *streamptr, long tsID, long recID, int memtype, void *data)
 {
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
-  int levelID = streamptr->tsteps[tsID].records[recID].levelID;
-  size_t gridsize = (size_t) gridInqSize(vlistInqVarGrid(streamptr->vlistID, varID));
+  recinfo_t *recInfo = &(streamptr->tsteps[tsID].recinfo[recID]);
+  size_t gridsize = (size_t) gridInqSize(vlistInqVarGrid(streamptr->vlistID, recInfo->varID));
 
   if (!data) data = Malloc(gridsize * ((memtype == MEMTYPE_FLOAT) ? sizeof(float) : sizeof(double)));
 
-  return (JobArgsCDF){
+  return (JobArgsCDF) {
     .streamptr = streamptr,
-    .varID = varID,
-    .levelID = levelID,
+    .varID = recInfo->varID,
+    .levelID = recInfo->levelID,
     .memtype = memtype,
     .recID = recID,
     .tsID = tsID,
@@ -887,7 +887,7 @@ cdf_get_local_step_and_recId(stream_t *streamptr, long globalRecId)
       globalRecId -= tsteps[1].nrecs;
     }
 
-  return (struct recTsId){ .recID = globalRecId, .tsID = localTsId };
+  return (struct recTsId) { .recID = globalRecId, .tsID = localTsId };
 }
 
 static void
diff --git a/src/cdf_records.c b/src/cdf_records.c
index 1ad97c2a7d8aa5f25b0fcf4306e68117ffabaa81..445c438c9a7d58979b4258507104e97e01f3cf83 100644
--- a/src/cdf_records.c
+++ b/src/cdf_records.c
@@ -16,7 +16,7 @@
 static void
 cdf_init_timestep(tsteps_t *timeStep, int numRecs, int numRecsAvail)
 {
-  timeStep->records = (record_t *) Malloc((size_t) numRecs * sizeof(record_t));
+  timeStep->recinfo = (recinfo_t *) Malloc((size_t) numRecs * sizeof(recinfo_t));
   timeStep->nrecs = numRecsAvail;
   timeStep->nallrecs = numRecs;
   timeStep->recordSize = numRecs;
@@ -39,7 +39,7 @@ cdf_get_numRecsAvail(int vlistID)
 }
 
 static void
-cdf_init_records_step0(int numRecs, int *recIDs, record_t *records, int vlistID)
+cdf_init_records_step0(int numRecs, int *recIDs, recinfo_t *recinfo, int vlistID)
 {
   for (int recID = 0; recID < numRecs; recID++) recIDs[recID] = recID;
 
@@ -50,20 +50,20 @@ cdf_init_records_step0(int numRecs, int *recIDs, record_t *records, int vlistID)
       int nlevels = zaxisInqSize(zaxisID);
       for (int levelID = 0; levelID < nlevels; levelID++)
         {
-          recordInitEntry(&records[recID]);
-          records[recID].varID = (short) varID;
-          records[recID].levelID = levelID;
+          recinfoInitEntry(&recinfo[recID]);
+          recinfo[recID].varID = (short) varID;
+          recinfo[recID].levelID = levelID;
           recID++;
         }
     }
 }
 
 static void
-cdf_init_records_step1(int numRecs, int *recIDs, record_t *records, int vlistID)
+cdf_init_records_step1(int numRecs, int *recIDs, recinfo_t *recinfo, int vlistID)
 {
   for (int recID = 0, vrecID = 0; recID < numRecs; recID++)
     {
-      if (vlistInqVarTimetype(vlistID, records[recID].varID) != TIME_CONSTANT)
+      if (vlistInqVarTimetype(vlistID, recinfo[recID].varID) != TIME_CONSTANT)
         {
           recIDs[vrecID++] = recID;
         }
@@ -94,7 +94,7 @@ cdf_create_records(stream_t *streamptr, size_t tsID)
       cdf_init_timestep(destTstep, numFields, numRecsAvail);
 
       destTstep->recIDs = (int *) Malloc((size_t) numRecsAvail * sizeof(int));
-      cdf_init_records_step0(numFields, destTstep->recIDs, destTstep->records, vlistID);
+      cdf_init_records_step0(numFields, destTstep->recIDs, destTstep->recinfo, vlistID);
     }
   else if (tsID == 1)
     {
@@ -104,17 +104,17 @@ cdf_create_records(stream_t *streamptr, size_t tsID)
 
       cdf_init_timestep(destTstep, numFields, numRecsAvail);
 
-      memcpy(destTstep->records, sourceTstep->records, (size_t) numFields * sizeof(record_t));
+      memcpy(destTstep->recinfo, sourceTstep->recinfo, (size_t) numFields * sizeof(recinfo_t));
 
       if (numRecsAvail)
         {
           destTstep->recIDs = (int *) Malloc((size_t) numRecsAvail * sizeof(int));
-          cdf_init_records_step1(numFields, destTstep->recIDs, destTstep->records, vlistID);
+          cdf_init_records_step1(numFields, destTstep->recIDs, destTstep->recinfo, vlistID);
         }
     }
   else
     {
-      if (streamptr->tsteps[1].records == 0) cdf_create_records(streamptr, 1);
+      if (streamptr->tsteps[1].recinfo == 0) cdf_create_records(streamptr, 1);
 
       int numRecsAvail = streamptr->tsteps[1].nrecs;
 
@@ -122,7 +122,7 @@ cdf_create_records(stream_t *streamptr, size_t tsID)
 
       cdf_init_timestep(destTstep, numFields, numRecsAvail);
 
-      memcpy(destTstep->records, sourceTstep->records, (size_t) numFields * sizeof(record_t));
+      memcpy(destTstep->recinfo, sourceTstep->recinfo, (size_t) numFields * sizeof(recinfo_t));
 
       if (numRecsAvail)
         {
diff --git a/src/cdf_util.c b/src/cdf_util.c
index 3f529d185cbfc8ddf7052a00993c8107885ad0f9..f4779e3f4d51e8a10f43af53571d7dfc54a90932 100644
--- a/src/cdf_util.c
+++ b/src/cdf_util.c
@@ -13,7 +13,6 @@
 #include <string.h>
 #include <ctype.h>
 
-#include "dmemory.h"
 #include "cdi.h"
 #include "cdi_int.h"
 #include "cdf_util.h"
diff --git a/src/cdf_util.h b/src/cdf_util.h
index 021f0467a32740a3c326272291a62dd21b9c24c6..8ac9d7158926d2150cdcc58b9fbba22c95366f72 100644
--- a/src/cdf_util.h
+++ b/src/cdf_util.h
@@ -1,6 +1,7 @@
 #ifndef CDF_UTIL_H_
 #define CDF_UTIL_H_
 
+#include <stdlib.h>
 #include <stdbool.h>
 
 bool xtypeIsText(int xtype);
diff --git a/src/cdf_write.c b/src/cdf_write.c
index eb556afcac38904a2ddbc11224283458fb6d6c16..4e5ece4fbaca9d08a7ee32710e8535ba80d2a990 100644
--- a/src/cdf_write.c
+++ b/src/cdf_write.c
@@ -13,7 +13,6 @@
 #include "cdf.h"
 #include "cdf_int.h"
 #include "vlist.h"
-#include "vlist_var.h"
 
 void
 cdfDefVarDeflate(int ncid, int ncvarID, int shuffle, int compLevel)
@@ -73,9 +72,10 @@ cdfDefVarSzip(int ncid, int ncvarID, int pixels_per_block)
 static nc_type
 cdfTypeComplexFloat(stream_t *streamptr)
 {
-  if (streamptr->nc_complex_float_id == CDI_UNDEFID)
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  if (cdfInfo->complexFloatId == CDI_UNDEFID)
     {
-      typedef struct complex_float
+      typedef struct
       {
         float r, i;
       } complex_float;
@@ -88,18 +88,19 @@ cdfTypeComplexFloat(stream_t *streamptr)
       if (status != NC_NOERR) Error("%s", nc_strerror(status));
       status = nc_insert_compound(fileID, nc_complex_id, "i", NC_COMPOUND_OFFSET(complex_float, i), NC_FLOAT);
       if (status != NC_NOERR) Error("%s", nc_strerror(status));
-      streamptr->nc_complex_float_id = nc_complex_id;
+      cdfInfo->complexFloatId = nc_complex_id;
     }
 
-  return (nc_type) streamptr->nc_complex_float_id;
+  return (nc_type) cdfInfo->complexFloatId;
 }
 
 static nc_type
 cdfTypeComplexDouble(stream_t *streamptr)
 {
-  if (streamptr->nc_complex_double_id == CDI_UNDEFID)
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  if (cdfInfo->complexDoubleId == CDI_UNDEFID)
     {
-      typedef struct complex_double
+      typedef struct
       {
         double r, i;
       } complex_double;
@@ -112,10 +113,10 @@ cdfTypeComplexDouble(stream_t *streamptr)
       if (status != NC_NOERR) Error("%s", nc_strerror(status));
       status = nc_insert_compound(fileID, nc_complex_id, "i", NC_COMPOUND_OFFSET(complex_double, i), NC_DOUBLE);
       if (status != NC_NOERR) Error("%s", nc_strerror(status));
-      streamptr->nc_complex_double_id = nc_complex_id;
+      cdfInfo->complexDoubleId = nc_complex_id;
     }
 
-  return (nc_type) streamptr->nc_complex_double_id;
+  return (nc_type) cdfInfo->complexDoubleId;
 }
 #endif
 
@@ -414,7 +415,7 @@ nc_grid_index(stream_t *streamptr, int gridID)
   int vlistID = streamptr->vlistID;
   int ngrids = vlistNumGrids(vlistID);
   for (index = 0; index < ngrids; ++index)
-    if (streamptr->ncgrid[index].gridID == gridID) break;
+    if (streamptr->cdfInfo.cdfGridVec[index].gridID == gridID) break;
 
   assert(index < ngrids);
 
@@ -577,9 +578,10 @@ cdfDefineCoordinates(const stream_t *streamptr, int ncvarID, int nczvarID, int g
       if (numLPE > 0) cdf_put_att_int(fileID, ncvarID, "CDI_grid_num_LPE", NC_INT, 1, &numLPE);
     }
 
+  const CdfGrid *cdfGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
   if (gridtype == GRID_GAUSSIAN_REDUCED)
     {
-      int ncyvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_Y];
+      int ncyvarID = cdfGrid->ncIdVec[CDF_VARID_Y];
       if (ncyvarID != CDI_UNDEFID)
         {
           char name[CDI_MAX_NAME];
@@ -589,7 +591,7 @@ cdfDefineCoordinates(const stream_t *streamptr, int ncvarID, int nczvarID, int g
           cdf_put_att_text(fileID, ncvarID, "CDI_grid_latitudes", len, name);
         }
 
-      int ncrpvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_RP];
+      int ncrpvarID = cdfGrid->ncIdVec[CDF_VARID_RP];
       if (ncrpvarID != CDI_UNDEFID)
         {
           char name[CDI_MAX_NAME];
@@ -613,25 +615,25 @@ cdfDefineCoordinates(const stream_t *streamptr, int ncvarID, int nczvarID, int g
     }
   else if (gridtype == GRID_LONLAT && xid == CDI_UNDEFID && yid == CDI_UNDEFID && gridsize == 1)
     {
-      int ncxvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_X];
-      int ncyvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_Y];
+      int ncxvarID = cdfGrid->ncIdVec[CDF_VARID_X];
+      int ncyvarID = cdfGrid->ncIdVec[CDF_VARID_Y];
       cdfAppendCoordinates(fileID, ncyvarID, coordinates);
       cdfAppendCoordinates(fileID, ncxvarID, coordinates);
     }
   else if (gridtype == GRID_GAUSSIAN_REDUCED)
     {
       /*
-      int ncxvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_X];
-      int ncyvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_Y];
+      int ncxvarID = ncgrid->ncIDs[CDF_VARID_X];
+      int ncyvarID = ncgrid->ncIDs[CDF_VARID_Y];
       cdfAppendCoordinates(fileID, ncyvarID, coordinates);
       cdfAppendCoordinates(fileID, ncxvarID, coordinates);
       */
     }
   else if (gridtype == GRID_UNSTRUCTURED || gridtype == GRID_CURVILINEAR)
     {
-      int ncxvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_X];
-      int ncyvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_Y];
-      int ncavarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_A];
+      int ncxvarID = cdfGrid->ncIdVec[CDF_VARID_X];
+      int ncyvarID = cdfGrid->ncIdVec[CDF_VARID_Y];
+      int ncavarID = cdfGrid->ncIdVec[CDF_VARID_A];
       // CMOR order: coordinates = "lat lon"
       if (CDI_Coordinates_Lon_Lat)
         {
@@ -671,12 +673,12 @@ cdfDefineCoordinates(const stream_t *streamptr, int ncvarID, int nczvarID, int g
     {
       if (gridInqXIsc(gridID))
         {
-          int ncxvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_X];
+          int ncxvarID = cdfGrid->ncIdVec[CDF_VARID_X];
           cdfAppendCoordinates(fileID, ncxvarID, coordinates);
         }
       else if (gridInqYIsc(gridID))
         {
-          int ncyvarID = streamptr->ncgrid[gridindex].ncIDs[CDF_VARID_Y];
+          int ncyvarID = cdfGrid->ncIdVec[CDF_VARID_Y];
           cdfAppendCoordinates(fileID, ncyvarID, coordinates);
         }
     }
@@ -702,19 +704,41 @@ calc_chunksize(size_t chunkSizeLim, size_t size)
 static const size_t chunkSizeMin = 262144;    // 256k
 static const size_t chunkSizeLim = 16777216;  // 16m
 
+size_t
+auto_chunksize_y(size_t gridsize, size_t xsize, size_t ysize)
+{
+  size_t gridChunkSize = calc_chunksize(chunkSizeLim, gridsize);
+  size_t yChunkSize = gridChunkSize / xsize;
+  size_t numChunks = (ysize % yChunkSize == 0) ? (ysize / yChunkSize) : (ysize / yChunkSize) + 1;
+  size_t modChunk = ysize % numChunks;
+  size_t maxChunks = numChunks * 2;
+  for (size_t i = numChunks; i < maxChunks; ++i)
+    {
+      size_t imod = ysize % i;
+      if (imod == 0 || imod > modChunk)
+        {
+          numChunks = i;
+          modChunk = imod;
+          if (modChunk == 0) break;
+        }
+    }
+  yChunkSize = ysize / numChunks;
+  return yChunkSize;
+}
+
 size_t
 calc_chunksize_y(int chunkType, size_t gridsize, size_t xsize, size_t ysize)
 {
   if (chunkType == CDI_CHUNK_AUTO)
-    return (gridsize <= chunkSizeMin) ? ysize : chunkSizeMin / xsize;
+    return auto_chunksize_y(gridsize, xsize, ysize);
   else
     return (chunkType == CDI_CHUNK_LINES) ? 1 : ysize;
 }
 
 size_t
-calc_chunksize_x(int chunkType, long chunkSize, size_t xsize, bool yIsUndefined)
+calc_chunksize_x(int chunkType, long chunkSize, size_t xsize, bool isReg2dGrid)
 {
-  if (chunkType == CDI_CHUNK_AUTO && yIsUndefined)
+  if (chunkType == CDI_CHUNK_AUTO && !isReg2dGrid)
     return (chunkSize > 0 && (size_t) chunkSize < xsize) ? (size_t) chunkSize : ((xsize <= chunkSizeMin) ? xsize : chunkSizeMin);
   else
     return calc_chunksize(chunkSizeLim, xsize);
@@ -722,8 +746,9 @@ calc_chunksize_x(int chunkType, long chunkSize, size_t xsize, bool yIsUndefined)
 
 static int
 cdfDefineDimsAndChunks(const stream_t *streamptr, int varID, int xid, int yid, int zid, size_t gridsize, const int dimorder[3],
-                       int dims[4], bool useChunks, size_t chunks[4], char axis[5], size_t *piax)
+                       int dims[4], bool useChunks, size_t chunks[4], char axis[5], size_t *piax, size_t *pchunkCacheSize)
 {
+  size_t chunkCacheSize = 1;
   int fileID = streamptr->fileID;
   int vlistID = streamptr->vlistID;
 
@@ -732,18 +757,22 @@ cdfDefineDimsAndChunks(const stream_t *streamptr, int varID, int xid, int yid, i
 
   for (int i = 0; i < 4; ++i) chunks[i] = 0;
 
-  size_t xsize = 0, ysize = 0;
+  size_t xsize = 0, ysize = 0, zsize = 0;
   if (xid != CDI_UNDEFID) cdf_inq_dimlen(fileID, xid, &xsize);
   if (yid != CDI_UNDEFID) cdf_inq_dimlen(fileID, yid, &ysize);
+  if (zid != CDI_UNDEFID) cdf_inq_dimlen(fileID, zid, &zsize);
 
   int timetype = vlistInqVarTimetype(vlistID, varID);
   if (vlistHasTime(vlistID) && timetype != TIME_CONSTANT)
     {
+      int chunkSizeDimT = 0;
+      cdiInqKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMT, &chunkSizeDimT);
       int tid = streamptr->basetime.ncdimid;
       if (tid == CDI_UNDEFID) Error("Internal problem, time undefined!");
       axis[iax++] = 'T';
-      chunks[ndims] = 1;
+      chunks[ndims] = (chunkSizeDimT > 0) ? chunkSizeDimT : 1;
       dims[ndims] = tid;
+      chunkCacheSize *= chunks[ndims];
       ndims++;
     }
 
@@ -753,9 +782,10 @@ cdfDefineDimsAndChunks(const stream_t *streamptr, int varID, int xid, int yid, i
   cdiInqKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE, &chunkSize);
   if (chunkSize > 0 && yid == CDI_UNDEFID) chunkType = CDI_CHUNK_AUTO;
 
-  if (chunkType == CDI_CHUNK_GRID && gridsize > ChunkSizeLim)
+  bool isReg2dGrid = (yid != CDI_UNDEFID && xid != CDI_UNDEFID);
+  if (chunkType == CDI_CHUNK_GRID && gridsize > ChunkSizeLim && isReg2dGrid)
     {
-      if (CDI_Debug) fprintf(stderr, "gridsize > %u, changed chunkType to CDI_CHUNK_LINES!\n", ChunkSizeLim);
+      if (CDI_Debug) fprintf(stderr, "gridsize > %u, changed chunkType to CDI_CHUNK_AUTO!\n", ChunkSizeLim);
       chunkType = CDI_CHUNK_AUTO;
     }
 
@@ -763,21 +793,35 @@ cdfDefineDimsAndChunks(const stream_t *streamptr, int varID, int xid, int yid, i
     {
       if (dimorder[id] == 3 && zid != CDI_UNDEFID)
         {
+          int chunkSizeDimZ = 0;
+          cdiInqKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMZ, &chunkSizeDimZ);
+          if (chunkSizeDimZ == -1) chunkSizeDimZ = zsize;
           axis[iax++] = 'Z';
-          chunks[ndims] = 1;
+          chunks[ndims] = (chunkSizeDimZ > 0) ? chunkSizeDimZ : 1;
           dims[ndims] = zid;
+          chunkCacheSize *= (chunkCacheSize > 1) ? zsize : chunks[ndims];
           ndims++;
         }
       else if (dimorder[id] == 2 && yid != CDI_UNDEFID)
         {
-          chunks[ndims] = calc_chunksize_y(chunkType, gridsize, xsize, ysize);
+          int chunkSizeDimY = 0;
+          cdiInqKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMY, &chunkSizeDimY);
+          if (chunkSizeDimY == -1) chunkSizeDimY = ysize;
+          if (chunkSizeDimY == 0) chunkSizeDimY = calc_chunksize_y(chunkType, gridsize, xsize, ysize);
+          chunks[ndims] = chunkSizeDimY;
           dims[ndims] = yid;
+          chunkCacheSize *= ysize;
           ndims++;
         }
       else if (dimorder[id] == 1 && xid != CDI_UNDEFID)
         {
-          chunks[ndims] = calc_chunksize_x(chunkType, chunkSize, xsize, (yid == CDI_UNDEFID));
+          int chunkSizeDimX = 0;
+          cdiInqKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMX, &chunkSizeDimX);
+          if (chunkSizeDimX == -1) chunkSizeDimX = xsize;
+          if (chunkSizeDimX == 0) chunkSizeDimX = calc_chunksize_x(chunkType, chunkSize, xsize, isReg2dGrid);
+          chunks[ndims] = chunkSizeDimX;
           dims[ndims] = xid;
+          chunkCacheSize *= xsize;
           ndims++;
         }
     }
@@ -786,6 +830,7 @@ cdfDefineDimsAndChunks(const stream_t *streamptr, int varID, int xid, int yid, i
     fprintf(stderr, "useChunks %d chunkType %d chunkSize %d  chunks %zu %zu %zu %zu\n", useChunks, chunkType, chunkSize, chunks[0],
             chunks[1], chunks[2], chunks[3]);
 
+  *pchunkCacheSize = chunkCacheSize;
   *piax = iax;
   return ndims;
 }
@@ -895,14 +940,15 @@ cdfGenVarname(int fileID, char name[CDI_MAX_NAME], int pnum, int pcat, int *pdis
 }
 
 static void
-cdfDefVarChunkCache(int fileID, int ncvarID)
+cdfDefVarChunkCache(int fileID, int ncvarID, size_t chunkCacheSize)
 {
   size_t size = 0;
   size_t nelems = 0;
   float preemption = 0;
-  if (CDI_Chunk_Cache > 0 && nc_get_var_chunk_cache(fileID, ncvarID, &size, &nelems, &preemption) == NC_NOERR)
+  if (nc_get_var_chunk_cache(fileID, ncvarID, &size, &nelems, &preemption) == NC_NOERR)
     {
-      size = (size_t) CDI_Chunk_Cache;
+      if (chunkCacheSize > size) size = chunkCacheSize;
+      if (CDI_Chunk_Cache > 0) size = (size_t) CDI_Chunk_Cache;
     }
 
   nc_set_var_chunk_cache(fileID, ncvarID, size, nelems, preemption);
@@ -927,14 +973,15 @@ cdfDefVar(stream_t *streamptr, int varID)
   SizeType gridsize = gridInqSize(gridID);
   int gridtype = gridInqType(gridID);
   int gridindex = nc_grid_index(streamptr, gridID);
-  int xid = (gridtype != GRID_TRAJECTORY) ? streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_X] : CDI_UNDEFID;
-  int yid = (gridtype != GRID_TRAJECTORY && gridtype != GRID_GAUSSIAN_REDUCED) ? streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_Y]
-                                                                               : CDI_UNDEFID;
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  const CdfGrid *cdfGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
+  int xid = (gridtype != GRID_TRAJECTORY) ? cdfGrid->ncIdVec[CDF_DIMID_X] : CDI_UNDEFID;
+  int yid = (gridtype != GRID_TRAJECTORY && gridtype != GRID_GAUSSIAN_REDUCED) ? cdfGrid->ncIdVec[CDF_DIMID_Y] : CDI_UNDEFID;
 
   int zaxisID = vlistInqVarZaxis(vlistID, varID);
   int zaxistype = zaxisInqType(zaxisID);
   int zaxisindex = vlistZaxisIndex(vlistID, zaxisID);
-  int zid = streamptr->zaxisID[zaxisindex];
+  int zid = cdfInfo->zaxisIdVec[zaxisindex];
 
   int dimorder[3];  // ZYX/321 and ZXY/312
   vlistInqVarDimorder(vlistID, varID, dimorder);
@@ -952,8 +999,9 @@ cdfDefVar(stream_t *streamptr, int varID)
   char axis[5];
   int dims[4];
   size_t chunks[4];
-  int ndims
-      = cdfDefineDimsAndChunks(streamptr, varID, xid, yid, zid, (size_t) gridsize, dimorder, dims, useChunks, chunks, axis, &iax);
+  size_t chunkCacheSize = 1;
+  int ndims = cdfDefineDimsAndChunks(streamptr, varID, xid, yid, zid, (size_t) gridsize, dimorder, dims, useChunks, chunks, axis,
+                                     &iax, &chunkCacheSize);
 
   char name[CDI_MAX_NAME];
   int length = CDI_MAX_NAME;
@@ -976,19 +1024,20 @@ cdfDefVar(stream_t *streamptr, int varID)
   else
     cdfGenVarname(fileID, name, pnum, pcat, &pdis, &code);
 
-  int dtype = vlistInqVarDatatype(vlistID, varID);
-  nc_type xtype = cdfDefDatatype(dtype, streamptr);
-
   if (streamptr->ncmode == 2)
     {
       cdf_redef(fileID);
       streamptr->ncmode = 1;
     }
 
+  int dtype = vlistInqVarDatatype(vlistID, varID);
+  nc_type xtype = cdfDefDatatype(dtype, streamptr);
+
   int ncvarID = -1;
   cdf_def_var(fileID, name, xtype, ndims, dims, &ncvarID);
 
-  cdfDefVarChunkCache(fileID, ncvarID);
+  chunkCacheSize *= cdf_xtype_to_numbytes(xtype);
+  cdfDefVarChunkCache(fileID, ncvarID, chunkCacheSize);
 
 #ifdef HAVE_NETCDF4
 #ifdef NC_QUANTIZE_BITROUND
@@ -1039,7 +1088,7 @@ cdfDefVar(stream_t *streamptr, int varID)
     }
 
   bool zaxisIsScalar = (zid == CDI_UNDEFID) ? (zaxisInqScalar(zaxisID) > 0) : false;
-  int nczvarID = (zaxisIsScalar || zaxistype == ZAXIS_CHAR) ? streamptr->nczvarID[zaxisindex] : CDI_UNDEFID;
+  int nczvarID = (zaxisIsScalar || zaxistype == ZAXIS_CHAR) ? streamptr->cdfInfo.ncZvarIdVec[zaxisindex] : CDI_UNDEFID;
 
   cdfDefineCoordinates(streamptr, ncvarID, nczvarID, gridtype, gridID, gridindex, xid, yid, (size_t) gridsize, axis, iax);
 
@@ -1106,8 +1155,9 @@ static void
 cdfWriteGridTraj(stream_t *streamptr, int gridID)
 {
   int gridindex = nc_grid_index(streamptr, gridID);
-  int lonID = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_X];
-  int latID = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_Y];
+  const CdfGrid *cdfGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
+  int lonID = cdfGrid->ncIdVec[CDF_DIMID_X];
+  int latID = cdfGrid->ncIdVec[CDF_DIMID_Y];
   size_t index = (size_t) streamptr->curTsID;
 
   double xlon = gridInqXval(gridID, 0);
@@ -1352,13 +1402,14 @@ cdfGetXYZid(stream_t *streamptr, int gridID, int zaxisID, int *xid, int *yid, in
   else
     {
       int gridindex = nc_grid_index(streamptr, gridID);
-      *xid = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_X];
-      if (gridtype != GRID_GAUSSIAN_REDUCED) *yid = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_Y];
+      const CdfGrid *cdfGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
+      *xid = cdfGrid->ncIdVec[CDF_DIMID_X];
+      if (gridtype != GRID_GAUSSIAN_REDUCED) *yid = cdfGrid->ncIdVec[CDF_DIMID_Y];
     }
 
   int vlistID = streamptr->vlistID;
   int zaxisindex = vlistZaxisIndex(vlistID, zaxisID);
-  *zid = streamptr->zaxisID[zaxisindex];
+  *zid = streamptr->cdfInfo.zaxisIdVec[zaxisindex];
 }
 
 static void
diff --git a/src/cdi.h b/src/cdi.h
index 624905c5dda1551c5803546224f815d35db79c47..d6e73b918c02aa5f7773a3d53ca10c19a0a23f8b 100644
--- a/src/cdi.h
+++ b/src/cdi.h
@@ -457,6 +457,10 @@ void    streamReadField(int streamID, double data[], SizeType *numMissVals);
 void    streamReadFieldF(int streamID, float data[], SizeType *numMissVals);
 void    streamCopyField(int streamIDdest, int streamIDsrc);
 
+void *  stream_get_pointer(int streamID);
+void *  stream_get_vlist_pointer(int streamID);
+void    pstreamInqField(void *streamPtr, int *varID, int *levelID);
+
 void    streamInqGRIBinfo(int streamID, int *intnum, float *fltnum, off_t *bignum);
 
 
@@ -695,6 +699,8 @@ int     vlistFindLevel(int vlistID, int fvarID, int flevelID);
 int     vlistMergedVar(int vlistID, int varID);
 int     vlistMergedLevel(int vlistID, int varID, int levelID);
 
+int     pvlistInqFlag(void *vlistPtr, int varID, int levelID);
+
 //      cdiClearAdditionalKeys: Clear the list of additional GRIB keys
 void    cdiClearAdditionalKeys(void);
 //      cdiDefAdditionalKey: Register an additional GRIB key which is read when file is opened
@@ -831,7 +837,6 @@ SizeType gridInqYCvals(int gridID, char *ycvals[]);
 #define  CDI_KEY_UNITS                         945  // Units of the variable
 #define  CDI_KEY_DATATYPE                      946  // Data type
 #define  CDI_KEY_REFERENCEURI                  947  // Reference URI to grid file
-#define  CDI_KEY_CHUNKS                        948  // Chunks
 
 // Integer keys
 #define  CDI_KEY_NUMBEROFGRIDUSED              961  // GRIB2 numberOfGridUsed
@@ -840,6 +845,10 @@ SizeType gridInqYCvals(int gridID, char *ycvals[]);
 #define  CDI_KEY_NLEV                          964  // GRIB2 nlev
 #define  CDI_KEY_CHUNKTYPE                     965  // ChunkType: CDI_CHUNK_AUTO/CDI_CHUNK_GRID/CDI_CHUNK_LINES
 #define  CDI_KEY_CHUNKSIZE                     966  // ChunkSize
+#define  CDI_KEY_CHUNKSIZE_DIMT                967  // ChunkSize time dimension
+#define  CDI_KEY_CHUNKSIZE_DIMZ                968  // ChunkSize zaxis dimension
+#define  CDI_KEY_CHUNKSIZE_DIMY                969  // ChunkSize yaxis dimension
+#define  CDI_KEY_CHUNKSIZE_DIMX                970  // ChunkSize xaxis dimension
 
 // Floating point keys
 #define  CDI_KEY_MISSVAL                       701  // Missing value
@@ -1306,6 +1315,10 @@ int  vlistInqVarTypeOfGeneratingProcess(int vlistID, int varID);
 void vlistDefVarTypeOfGeneratingProcess(int vlistID, int varID, int typeOfGeneratingProcess);
 void vlistDefVarProductDefinitionTemplate(int vlistID, int varID, int productDefinitionTemplate);
 
+// Compatibility functions for ParaView vtkCDIReader (obsolete functions)
+int vlistNgrids(int vlistID); // calls vlistNumGrids()
+int vlistNzaxis(int vlistID); // calls vlistNumZaxis()
+
 #ifdef __cplusplus
 }
 #endif
@@ -1437,6 +1450,9 @@ extern int (*proj_stere_to_lonlat_func)(struct CDI_GridProjParams gpp, double, d
 // Used on CDO remap_scrip_io.cc
 void cdf_def_var_filter(int ncid, int ncvarID, const char *filterSpec);
 
+int cdi_has_dap(void);
+int cdi_has_cgribex(void);
+
 #ifdef __cplusplus
 }
 #endif
diff --git a/src/cdi_fdb.h b/src/cdi_fdb.h
index 9baf530494deeef70c26ad2f78643cc96922f856..1d3549c761a33c7e74019b5e896a7639ae605cb6 100644
--- a/src/cdi_fdb.h
+++ b/src/cdi_fdb.h
@@ -5,8 +5,6 @@
 #include "config.h"
 #endif
 
-extern int cdi_fdb_dummy;
-
 #ifdef HAVE_LIBFDB5
 
 #include <fdb5/api/fdb_c.h>
diff --git a/src/cdi_int.c b/src/cdi_int.c
index 347090c20199a71e38299791856b2121a6b835c1..04ed90af6d313018a152d40efb0c18c4570c52fa 100644
--- a/src/cdi_int.c
+++ b/src/cdi_int.c
@@ -41,12 +41,15 @@ int CDI_Inventory_Mode = 1;
 int CDI_Version_Info = 1;
 int CDI_Query_Abort = 1;
 int CDI_Convert_Cubesphere = 1;
+int CDI_Read_Cell_Center = 1;
 int CDI_Read_Cell_Corners = 1;
 int CDI_CMOR_Mode = 0;
 int CDI_Reduce_Dim = 0;
 int CDI_Shuffle = 0;
 int CDI_Test = 0;
 size_t CDI_Netcdf_Hdr_Pad = 0UL;
+bool CDI_CopyChunkSpec = false;
+bool CDI_RemoveChunkSpec = false;
 bool CDI_Chunk_Cache_Info = false;
 long CDI_Chunk_Cache = -1L;
 size_t CDI_Chunk_Cache_Max = 0UL;
@@ -347,6 +350,9 @@ cdiInitialize(void)
       value = cdi_getenv_int("CDI_LOCK_IO");
       if (value >= 0) CDI_Lock_IO = (bool) value;
 
+      value = cdi_getenv_int("CDI_READ_CELL_CENTER");
+      if (value >= 0) CDI_Read_Cell_Center = (int) value;
+
       value = cdi_getenv_int("CDI_READ_CELL_CORNERS");
       if (value >= 0) CDI_Read_Cell_Corners = (int) value;
 
@@ -368,6 +374,12 @@ cdiInitialize(void)
       value = cdi_getenv_int("CDI_NETCDF_HDR_PAD");
       if (value >= 0) CDI_Netcdf_Hdr_Pad = (size_t) value;
 
+      value = cdi_getenv_int("CDI_COPY_CHUNKSPEC");
+      if (value >= 0) CDI_CopyChunkSpec = (value > 0);
+
+      value = cdi_getenv_int("CDI_REMOVE_CHUNKSPEC");
+      if (value >= 0) CDI_RemoveChunkSpec = (value > 0);
+
       value = cdi_getenv_int("CDI_CHUNK_CACHE_INFO");
       if (value > 0) CDI_Chunk_Cache_Info = true;
 
@@ -515,11 +527,14 @@ cdiDefGlobal(const char *string, int value)
   else if (str_is_equal(string, "SORTNAME"))              cdiSortName = value;
   else if (str_is_equal(string, "HAVE_MISSVAL"))          cdiHaveMissval = value;
   else if (str_is_equal(string, "NC_CHUNKSIZEHINT"))      CDI_Netcdf_Chunksizehint = value;
+  else if (str_is_equal(string, "READ_CELL_CENTER"))      CDI_Read_Cell_Center = value;
   else if (str_is_equal(string, "READ_CELL_CORNERS"))     CDI_Read_Cell_Corners = value;
   else if (str_is_equal(string, "CMOR_MODE"))             CDI_CMOR_Mode = value;
   else if (str_is_equal(string, "REDUCE_DIM"))            CDI_Reduce_Dim = value;
   else if (str_is_equal(string, "NETCDF_HDR_PAD"))        CDI_Netcdf_Hdr_Pad = (size_t) value;
   else if (str_is_equal(string, "NETCDF_LAZY_GRID_LOAD")) CDI_Netcdf_Lazy_Grid_Load = (bool) value;
+  else if (str_is_equal(string, "COPY_CHUNKSPEC"))        CDI_CopyChunkSpec = (bool) value;
+  else if (str_is_equal(string, "REMOVE_CHUNKSPEC"))      CDI_RemoveChunkSpec = (bool) value;
   else Warning("Unsupported global key: %s", string);
   // clang-format on
 }
@@ -564,6 +579,35 @@ cdiBaseFiletype(int filetype)
   return filetype;
 }
 
+void
+stream_def_accesstype(stream_t *s, int type)
+{
+  if (s->accesstype == CDI_UNDEFID)
+    {
+      s->accesstype = type;
+    }
+  else if (s->accesstype != type)
+    Error("Changing access type from %s not allowed!", s->accesstype == TYPE_REC ? "REC to VAR" : "VAR to REC");
+}
+
+int
+cdi_has_dap(void)
+{
+#ifdef HAVE_LIBNC_DAP
+  return 1;
+#endif
+  return 0;
+}
+
+int
+cdi_has_cgribex(void)
+{
+#ifdef HAVE_LIBCGRIBEX
+  return 1;
+#endif
+  return 0;
+}
+
 /*
  * Local Variables:
  * c-file-style: "Java"
diff --git a/src/cdi_int.h b/src/cdi_int.h
index 847b95d36ee2e02b362cba2e4253ae8c82086fc3..f85c782869ee942298ee41afa3da642ad4d16018 100644
--- a/src/cdi_int.h
+++ b/src/cdi_int.h
@@ -171,23 +171,31 @@ varScanKeysIsEqual(const VarScanKeys *s1, const VarScanKeys *s2)
   return memcmp(s1, s2, sizeof(VarScanKeys)) == 0;
 }
 
+typedef struct
+{
+  int levelID;
+  short varID;
+  short used;
+} recinfo_t;
+
 typedef struct
 {
   off_t position;
   size_t size;
   size_t gridsize;
-  int zip;
   int param;
   int ilevel;
   int ilevel2;
-  int ltype;
+  short ltype;
   short tsteptype;
-  short varID;
-  int levelID;
-  short used;
-  char varname[32];  // needed for grib decoding with GRIB_API
+#ifdef HAVE_LIBGRIB
+  int zip;
   VarScanKeys scanKeys;
   var_tile_t tiles;  // tile-related meta-data, currently for GRIB-API only.
+#ifdef HAVE_LIBGRIB_API
+  char varname[32];
+#endif
+#endif
 #ifdef HAVE_LIBFDB5
   int fdbItemIndex;
 #endif
@@ -196,6 +204,7 @@ typedef struct
 typedef struct
 {
   int *recIDs;  // IDs of non constant records
+  recinfo_t *recinfo;
   record_t *records;
   int recordSize;   // number of allocated records
   int nrecs;        // number of used records
@@ -250,16 +259,31 @@ enum cdfIDIdx
   CDF_VARID_Y,
   CDF_VARID_RP,  // reducedPoints
   CDF_VARID_A,
-  CDF_SIZE_ncIDs,
+  CDF_SIZE_NCID,
 };
 
 typedef struct
 {
-  int ncIDs[CDF_SIZE_ncIDs];
+  int ncIdVec[CDF_SIZE_NCID];
   int gridID;
   long start;
   long count;
-} ncgrid_t;
+} CdfGrid;
+
+typedef struct
+{
+  int complexFloatId;
+  int complexDoubleId;
+  CdfGrid cdfGridVec[MAX_GRIDS_PS];
+  int zaxisIdVec[MAX_ZAXES_PS];  // Warning: synchronous array to vlist_to_pointer(vlistID)->zaxisIDs
+  int ncZvarIdVec[MAX_ZAXES_PS];
+  int ncDimIdVec[MAX_DIMS_PS];
+  size_t ncDimLenVec[MAX_DIMS_PS];
+  int ncNumDims;
+  size_t chunkSizeDimT;
+  size_t chunkSizeDimZ;
+  VCT vct;
+} CdfInfo;
 #endif
 
 typedef struct
@@ -290,17 +314,7 @@ typedef struct
   int ncmode;
   int vlistID;
 #ifdef HAVE_LIBNETCDF
-  int nc_complex_float_id;
-  int nc_complex_double_id;
-  ncgrid_t ncgrid[MAX_GRIDS_PS];
-  int zaxisID[MAX_ZAXES_PS];  // Warning: synchronous array to vlist_to_pointer(vlistID)->zaxisIDs
-  int nczvarID[MAX_ZAXES_PS];
-  int ncNumDims;
-  int ncDimID[MAX_DIMS_PS];
-  size_t ncDimLen[MAX_DIMS_PS];
-  VCT vct;
-  size_t chunkSizeTdim;
-  size_t chunkSizeZdim;
+  CdfInfo cdfInfo;
 #endif
   long maxGlobalRecs;
   int globalatts;
@@ -403,11 +417,14 @@ extern int CDI_Inventory_Mode;
 extern int CDI_Query_Abort;
 extern int CDI_Version_Info;
 extern int CDI_Convert_Cubesphere;
+extern int CDI_Read_Cell_Center;
 extern int CDI_Read_Cell_Corners;
 extern int CDI_CMOR_Mode;
 extern int CDI_Reduce_Dim;
 extern int CDI_Shuffle;
 extern size_t CDI_Netcdf_Hdr_Pad;
+extern bool CDI_CopyChunkSpec;
+extern bool CDI_RemoveChunkSpec;
 extern bool CDI_Chunk_Cache_Info;
 extern long CDI_Chunk_Cache;
 extern size_t CDI_Chunk_Cache_Max;
@@ -455,7 +472,7 @@ void cdi_generate_vars(stream_t *streamptr);
 
 void vlist_check_contents(int vlistID);
 
-void cdi_create_records(stream_t *streamptr, int tsID);
+void cdi_create_records(stream_t *streamptr, int tsID, bool allocRecords);
 
 void streamFCopyRecord(stream_t *streamptr2, stream_t *streamptr1, const char *container_name);
 
@@ -463,12 +480,11 @@ int recordNewEntry(stream_t *streamptr, int tsID);
 
 void cdi_create_timesteps(size_t numTimesteps, stream_t *streamptr);
 
-void recordInitEntry(record_t *record);
+void recinfoInitEntry(recinfo_t *recinfo);
 
 void cdiCheckZaxis(int zaxisID);
 
-void cdiDefAccesstype(int streamID, int type);
-int cdiInqAccesstype(int streamID);
+void stream_def_accesstype(stream_t *s, int type);
 
 int getByteswap(int byteorder);
 
diff --git a/src/cdi_key.c b/src/cdi_key.c
index e5c00dc940f93e8d4787300d09139a28c98f4648..3f0351a1e366d8eac9054cf0e10e30d82c3c6abe 100644
--- a/src/cdi_key.c
+++ b/src/cdi_key.c
@@ -12,6 +12,7 @@
 
 #include "cdi.h"
 #include "cdi_int.h"
+#include "taxis.h"
 #include "zaxis.h"
 #include "grid.h"
 #include "vlist.h"
@@ -43,6 +44,12 @@ zaxis_get_keysp(zaxis_t *zaxisptr, int varID)
   return (varID == CDI_GLOBAL) ? &zaxisptr->keys : NULL;
 }
 
+static cdi_keys_t *
+taxis_get_keysp(taxis_t *taxisptr, int varID)
+{
+  return (varID == CDI_GLOBAL) ? &taxisptr->keys : NULL;
+}
+
 static cdi_key_t *
 new_key(cdi_keys_t *keysp, int key)
 {
@@ -68,7 +75,7 @@ find_key(cdi_keys_t *keysp, int key)
 
   if (keysp->nelems == 0) return NULL;
 
-  for (size_t keyid = 0; keyid < keysp->nelems; keyid++)
+  for (uint16_t keyid = 0; keyid < keysp->nelems; keyid++)
     {
       cdi_key_t *keyp = &(keysp->value[keyid]);
       if (keyp->key == key) return keyp;  // Normal return
@@ -84,7 +91,7 @@ find_key_const(const cdi_keys_t *keysp, int key)
 
   if (keysp->nelems == 0) return NULL;
 
-  for (size_t keyid = 0; keyid < keysp->nelems; keyid++)
+  for (uint16_t keyid = 0; keyid < keysp->nelems; keyid++)
     {
       const cdi_key_t *keyp = &(keysp->value[keyid]);
       if (keyp->key == key) return keyp;  // Normal return
@@ -96,10 +103,12 @@ find_key_const(const cdi_keys_t *keysp, int key)
 static cdi_keys_t *
 cdi_get_keysp(int objID, int varID)
 {
-  if (reshGetTxCode(objID) == GRID) return grid_get_keysp(grid_to_pointer(objID), varID);
-  if (reshGetTxCode(objID) == DIST_GRID) return grid_get_keysp(grid_to_pointer(objID), varID);
-  if (reshGetTxCode(objID) == ZAXIS) return zaxis_get_keysp(zaxis_to_pointer(objID), varID);
-  if (reshGetTxCode(objID) == VLIST) return vlist_get_keysp(vlist_to_pointer(objID), varID);
+  int reshID = reshGetTxCode(objID);
+  if (reshID == GRID) return grid_get_keysp(grid_to_pointer(objID), varID);
+  if (reshID == DIST_GRID) return grid_get_keysp(grid_to_pointer(objID), varID);
+  if (reshID == ZAXIS) return zaxis_get_keysp(zaxis_to_pointer(objID), varID);
+  if (reshID == TAXIS) return taxis_get_keysp(taxis_to_pointer(objID), varID);
+  if (reshID == VLIST) return vlist_get_keysp(vlist_to_pointer(objID), varID);
 
   return NULL;
 }
@@ -114,9 +123,9 @@ cdi_key_compare(cdi_keys_t *keyspa, cdi_keys_t *keyspb, int keynum)
   if (keypa->type != keypb->type) return 1;
   if (keypa->length != keypb->length) return 1;
 
-  if (keypa->type == KEY_BYTES) return (memcmp(keypa->v.s, keypb->v.s, (size_t) keypa->length) != 0);
-  if (keypa->type == KEY_FLOAT) return (IS_NOT_EQUAL(keypa->v.d, keypb->v.d));
-  if (keypa->type == KEY_INT) return (keypa->v.i != keypb->v.i);
+  if (keypa->type == KeyBytes) return (memcmp(keypa->v.s, keypb->v.s, (size_t) keypa->length) != 0);
+  if (keypa->type == KeyFloat) return (IS_NOT_EQUAL(keypa->v.d, keypb->v.d));
+  if (keypa->type == KeyInt) return (keypa->v.i != keypb->v.i);
 
   return 0;
 }
@@ -127,16 +136,16 @@ cdi_delete_key(cdi_key_t *keyp)
   if (keyp != NULL && keyp->length)  // key in use
     {
       keyp->length = 0;
-      if (keyp->type == KEY_BYTES)
+      if (keyp->type == KeyBytes)
         {
           if (keyp->v.s) Free(keyp->v.s);
           keyp->v.s = NULL;
         }
-      else if (keyp->type == KEY_FLOAT)
+      else if (keyp->type == KeyFloat)
         {
           keyp->v.d = 0.0;
         }
-      else if (keyp->type == KEY_INT)
+      else if (keyp->type == KeyInt)
         {
           keyp->v.i = 0;
         }
@@ -146,8 +155,8 @@ cdi_delete_key(cdi_key_t *keyp)
 void
 cdiDeleteVarKeys(cdi_keys_t *keysp)
 {
-  int nelems = keysp ? (int) keysp->nelems : 0;
-  for (int keyid = 0; keyid < nelems; keyid++)
+  uint16_t nelems = keysp ? keysp->nelems : 0;
+  for (uint16_t keyid = 0; keyid < nelems; keyid++)
     {
       cdi_delete_key(&(keysp->value[keyid]));
     }
@@ -167,20 +176,20 @@ cdiDeleteKeys(int cdiID, int varID)
 void
 cdiPrintVarKeys(cdi_keys_t *keysp)
 {
-  int nelems = keysp ? (int) keysp->nelems : 0;
-  for (int keyid = 0; keyid < nelems; keyid++)
+  uint16_t nelems = keysp ? (int) keysp->nelems : 0;
+  for (uint16_t keyid = 0; keyid < nelems; keyid++)
     {
       cdi_key_t *keyp = &(keysp->value[keyid]);
       if (keyp->length == 0) continue;
-      if (keyp->type == KEY_BYTES)
+      if (keyp->type == KeyBytes)
         {
           fprintf(stdout, "%d key %d length %d value %s\n", keyid + 1, keyp->key, keyp->length, keyp->v.s);
         }
-      else if (keyp->type == KEY_FLOAT)
+      else if (keyp->type == KeyFloat)
         {
           fprintf(stdout, "%d key %d value %g\n", keyid + 1, keyp->key, keyp->v.d);
         }
-      else if (keyp->type == KEY_INT)
+      else if (keyp->type == KeyInt)
         {
           fprintf(stdout, "%d key %d value %d\n", keyid + 1, keyp->key, keyp->v.i);
         }
@@ -219,9 +228,9 @@ static void
 cdi_define_key(const cdi_key_t *keyp, cdi_keys_t *keysp)
 {
   // clang-format off
-  if      (keyp->type == KEY_INT)   cdiDefVarKeyInt(keysp, keyp->key, keyp->v.i);
-  else if (keyp->type == KEY_FLOAT) cdiDefVarKeyFloat(keysp, keyp->key, keyp->v.d);
-  else if (keyp->type == KEY_BYTES) cdiDefVarKeyBytes(keysp, keyp->key, keyp->v.s, keyp->length);
+  if      (keyp->type == KeyInt)   cdiDefVarKeyInt(keysp, keyp->key, keyp->v.i);
+  else if (keyp->type == KeyFloat) cdiDefVarKeyFloat(keysp, keyp->key, keyp->v.d);
+  else if (keyp->type == KeyBytes) cdiDefVarKeyBytes(keysp, keyp->key, keyp->v.s, keyp->length);
   // clang-format on
 }
 
@@ -241,7 +250,7 @@ cdiDeleteKey(int cdiID, int varID, int key)
 void
 cdiCopyVarKeys(const cdi_keys_t *keysp1, cdi_keys_t *keysp2)
 {
-  for (size_t keyid = 0; keyid < keysp1->nelems; keyid++)
+  for (uint16_t keyid = 0; keyid < keysp1->nelems; keyid++)
     {
       const cdi_key_t *keyp = &(keysp1->value[keyid]);
       if (keyp->length > 0) cdi_define_key(keyp, keysp2);
@@ -299,7 +308,7 @@ cdiDefVarKeyInt(cdi_keys_t *keysp, int key, int value)
     {
       // if ( keyp->v.i != value )
       {
-        keyp->type = KEY_INT;
+        keyp->type = KeyInt;
         keyp->v.i = value;
         keyp->length = 1;
       }
@@ -370,7 +379,7 @@ cdiInqKeyInt(int cdiID, int varID, int key, int *value)
   const cdi_key_t *keyp = find_key_const(keysp, key);
   if (keyp != NULL && keyp->length == 1)  // key in use
     {
-      if (keyp->type == KEY_INT)
+      if (keyp->type == KeyInt)
         {
           *value = keyp->v.i;
           status = CDI_NOERR;
@@ -386,7 +395,7 @@ cdiInqVarKeyInt(const cdi_keys_t *keysp, int key)
   int value = 0;
 
   const cdi_key_t *keyp = find_key_const(keysp, key);
-  if (keyp && keyp->type == KEY_INT) value = keyp->v.i;
+  if (keyp && keyp->type == KeyInt) value = keyp->v.i;
 
   return value;
 }
@@ -399,7 +408,7 @@ cdiDefVarKeyFloat(cdi_keys_t *keysp, int key, double value)
 
   if (keyp != NULL)
     {
-      keyp->type = KEY_FLOAT;
+      keyp->type = KeyFloat;
       keyp->v.d = value;
       keyp->length = 1;
     }
@@ -469,7 +478,7 @@ cdiInqKeyFloat(int cdiID, int varID, int key, double *value)
   const cdi_key_t *keyp = find_key_const(keysp, key);
   if (keyp != NULL && keyp->length == 1)  // key in use
     {
-      if (keyp->type == KEY_FLOAT)
+      if (keyp->type == KeyFloat)
         {
           *value = keyp->v.d;
           status = CDI_NOERR;
@@ -500,7 +509,7 @@ cdiDefVarKeyBytes(cdi_keys_t *keysp, int key, const unsigned char *bytes, int le
         }
 
       memcpy(keyp->v.s, bytes, length_);
-      keyp->type = KEY_BYTES;
+      keyp->type = KeyBytes;
     }
 }
 
@@ -546,7 +555,7 @@ cdiInqVarKeyBytes(const cdi_keys_t *keysp, int key, unsigned char *bytes, int *l
   int val_len;
   if (keyp != NULL && (val_len = keyp->length) > 0)  // key in use
     {
-      if (keyp->type == KEY_BYTES)
+      if (keyp->type == KeyBytes)
         {
           if (val_len < *length)
             *length = val_len;
@@ -702,7 +711,7 @@ cdiInqVarKeyStringPtr(const cdi_keys_t *keysp, int key)
   const cdi_key_t *keyp = find_key_const(keysp, key);
   if (keyp != NULL)  // key in use
     {
-      if (keyp->type == KEY_BYTES) return (const char *) keyp->v.s;
+      if (keyp->type == KeyBytes) return (const char *) keyp->v.s;
     }
 
   return NULL;
@@ -713,7 +722,7 @@ cdiInitKeys(cdi_keys_t *keysp)
 {
   keysp->nalloc = MAX_KEYS;
   keysp->nelems = 0;
-  for (int i = 0; i < MAX_KEYS; ++i) keysp->value[i].length = 0;
+  for (uint16_t i = 0; i < MAX_KEYS; ++i) keysp->value[i].length = 0;
 }
 
 /*
diff --git a/src/cdi_key.h b/src/cdi_key.h
index 71755838c36dd14983b189ceeada51516a09a59c..8fa4f03637dff751c1ab7aba7bb29543eff38bb8 100644
--- a/src/cdi_key.h
+++ b/src/cdi_key.h
@@ -2,14 +2,15 @@
 #define CDI_KEY_H
 
 #include <stdio.h>
+#include <stdint.h>
 #include "cdi_limits.h"
 
 // CDI key
 typedef struct
 {
-  int key;     // CDI key
-  int type;    // KEY_INT, KEY_FLOAT, KEY_BYTES
-  int length;  // number of bytes in v.s
+  uint16_t key;   // CDI key
+  uint16_t type;  // KEY_INT, KEY_FLOAT, KEY_BYTES
+  int length;     // number of bytes in v.s
   union
   {
     int i;
@@ -20,16 +21,16 @@ typedef struct
 
 typedef struct
 {
-  size_t nalloc;  // number allocated >= nelems
-  size_t nelems;  // length of the array
+  uint16_t nalloc;  // number allocated >= nelems
+  uint16_t nelems;  // length of the array
   cdi_key_t value[MAX_KEYS];
 } cdi_keys_t;
 
 enum
 {
-  KEY_INT = 1,
-  KEY_FLOAT,
-  KEY_BYTES
+  KeyInt = 1,
+  KeyFloat,
+  KeyBytes
 };
 
 void cdiDefVarKeyInt(cdi_keys_t *keysp, int key, int value);
diff --git a/src/cdi_util.c b/src/cdi_util.c
index 187cf2c530db3c7c1d9229384cfcf5c050d74ca1..1609557054206213c43706fa1877e3af3bcef7f8 100644
--- a/src/cdi_util.c
+++ b/src/cdi_util.c
@@ -2,8 +2,9 @@
 #include "config.h"
 #endif
 
+#ifndef _WIN32
 #include <unistd.h>
-
+#endif
 #include <limits.h>
 #include <stdio.h>
 #include <stdlib.h>
diff --git a/src/cdi_uuid.h b/src/cdi_uuid.h
index e55d3a8959c4abd9c4393d1367ee78eddd3b4dc3..cbb3071899d9434c23eb65d195c90400b4863811 100644
--- a/src/cdi_uuid.h
+++ b/src/cdi_uuid.h
@@ -24,7 +24,9 @@ int cdiUUIDIsNull(const unsigned char uuid[])
   return isNull;
 }
 
+#ifndef _WIN32
 void cdiCreateUUID(unsigned char uuid[CDI_UUID_SIZE]);
+#endif
 
 int cdiUUID2Str(const unsigned char uuid[], char uuidstr[]);
 int cdiStr2UUID(const char *uuidstr, unsigned char uuid[]);
diff --git a/src/cmake/cdi/cdi-config.cmake.in b/src/cmake/cdi/cdi-config.cmake.in
index e34cb86e5e2ca978fc9b156b5277ba1b2eaa8502..f2f972547906a2212567d9e37c5b8461363df4b6 100644
--- a/src/cmake/cdi/cdi-config.cmake.in
+++ b/src/cmake/cdi/cdi-config.cmake.in
@@ -1,9 +1,9 @@
 # Config file for lib CDI project
 # run cmake with CDI_DIR pointing to the directory containing this file.
 
-# set(CDI_ROOT_DIR "${CMAKE_CURRENT_LIST_DIR}/../../..")
+set(CDI_ROOT_DIR "${CMAKE_CURRENT_LIST_DIR}/../../..")
 # use following line instead of previous if this file can be configured 
-set(CDI_ROOT_DIR "@prefix@")
+# set(CDI_ROOT_DIR "@prefix@")
 
 find_path(CDI_INCLUDE_DIRECTORY
   cdi.h
@@ -15,7 +15,7 @@ find_path(CDI_INCLUDE_DIRECTORY
 mark_as_advanced(CDI_INCLUDE_DIRECTORY)
 
 find_library(CDI_LIBRARY
-  NAMES cdi
+  NAMES cdilib
   PATHS "${CDI_ROOT_DIR}/lib"
   # use following line instead of previous if this file can be configured 
   # PATHS "@libdir@"
diff --git a/src/extra.h b/src/extra.h
index 1c586432db021af78776419ad4debe01e81ffbef..0c046b399015b44b8c738da5272e2d283d5ff81b 100644
--- a/src/extra.h
+++ b/src/extra.h
@@ -5,6 +5,8 @@
 #include "config.h"
 #endif
 
+#include <stdlib.h>
+
 enum
 {
   EXT_REAL = 1,
diff --git a/src/extralib.c b/src/extralib.c
index cf3b0a72645b35cc23648603df86b7e9fd25230a..0bcc85454fd26586f1f8393b87ff439a4e08700a 100644
--- a/src/extralib.c
+++ b/src/extralib.c
@@ -11,7 +11,6 @@
  * next to find out what happened */
 
 #include <stdio.h>
-#include <stdlib.h>
 #include <string.h>
 #include <ctype.h>
 
diff --git a/src/file.c b/src/file.c
index 2086da8a5643a461bf96f56909a7f5dba5fc8578..d38c5a2516b3955fbb13bad9f8553f82e8553b2b 100644
--- a/src/file.c
+++ b/src/file.c
@@ -10,19 +10,25 @@
 #include "config.h"
 #endif
 
-#include <unistd.h>
-
 #include <assert.h>
 #include <ctype.h>
 #include <errno.h>
 #include <fcntl.h>
 #include <limits.h>
+#include <stdbool.h>
 #include <stdio.h>
 #include <stdlib.h>
-#include <stdbool.h>
 #include <string.h>
-#include <sys/types.h>
 #include <sys/stat.h>
+#include <sys/types.h>
+
+// On Windows, define ssize_t manually
+#ifdef _WIN32
+#define ssize_t __int64
+#include <io.h>
+#else
+#include <unistd.h>
+#endif
 
 #ifdef HAVE_SYS_TIME_H
 #include <sys/time.h>  // gettimeofday()
@@ -392,7 +398,7 @@ fileFlush(int fileID)
     {
       FILE *fp = fileptr->fp;
       retval = fflush(fp);
-      if (retval == 0) retval = fsync(fileno(fp));
+      if (retval == 0) retval = fflush(fp);
       if (retval != 0) retval = errno;
     }
 
diff --git a/src/gaussian_latitudes.c b/src/gaussian_latitudes.c
index 7196999fc060af391969c8193574959dbaa89ed1..f62954c319556df722c1c02897dca50f86cf7d39 100644
--- a/src/gaussian_latitudes.c
+++ b/src/gaussian_latitudes.c
@@ -14,6 +14,9 @@
 #include <stdlib.h>
 #include <stdbool.h>
 #include <float.h>
+
+// Required on windows to be able to use M_PI
+#define _USE_MATH_DEFINES
 #include <math.h>
 
 #ifndef M_SQRT2
diff --git a/src/grb_read.c b/src/grb_read.c
index 9c82461945d496e7b14a77979cd8bda2addf1717..5868c1618e73bc7a0a74a27333f6ac270a8bdf0d 100644
--- a/src/grb_read.c
+++ b/src/grb_read.c
@@ -111,7 +111,8 @@ grib1_unzip_record(void *gribbuffer, size_t *gribsize)
 
 typedef struct JobArgsGRB
 {
-  int recID, tsID, *outZip, filetype, memType, datatype, unreduced;
+  int *outZip;
+  int recID, tsID, filetype, memType, datatype, unreduced;
   void *cgribexp, *gribbuffer, *data;
   size_t recsize, gridsize, numMissVals;
   double missval;
@@ -131,7 +132,7 @@ static JobArgsGRB
 grb_read_raw_data(stream_t *streamptr, int tsID, int recID, int memType, void *gribbuffer, void *data, bool resetFilePos)
 {
   int vlistID = streamptr->vlistID;
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
+  int varID = streamptr->tsteps[tsID].recinfo[recID].varID;
   size_t recsize = streamptr->tsteps[tsID].records[recID].size;
 
   int gridID = vlistInqVarGrid(vlistID, varID);
@@ -167,7 +168,7 @@ grb_read_raw_data(stream_t *streamptr, int tsID, int recID, int memType, void *g
       if (!resetFilePos) streamptr->numvals += (SizeType) gridsize;
     }
 
-  return (JobArgsGRB){
+  return (JobArgsGRB) {
     .recID = recID,
     .tsID = tsID,
     .outZip = &streamptr->tsteps[tsID].records[recID].zip,
diff --git a/src/grb_write.c b/src/grb_write.c
index 9ece9e4c10669e5875cc02efa512ac4047b1c8d8..217904313fda17b3dfc7d6f1a7d3caf7ed14f976 100644
--- a/src/grb_write.c
+++ b/src/grb_write.c
@@ -224,7 +224,8 @@ grbCopyField(stream_t *streamptr2, stream_t *streamptr1)
   int tsID = streamptr1->curTsID;
   int vrecID = streamptr1->tsteps[tsID].curRecID;
   int recID = streamptr1->tsteps[tsID].recIDs[vrecID];
-  const record_t *record = &streamptr1->tsteps[tsID].records[recID];
+  const recinfo_t *recinfo = &(streamptr1->tsteps[tsID].recinfo[recID]);
+  const record_t *record = &(streamptr1->tsteps[tsID].records[recID]);
   off_t recpos = record->position;
   size_t recsize = record->size;
 
@@ -347,8 +348,7 @@ grbCopyField(stream_t *streamptr2, stream_t *streamptr1)
   if (streamptr2->protocol == CDI_PROTOCOL_FDB)
     {
       int vlistID = streamptr1->vlistID;
-      int varID = record->varID;
-      int zaxisID = vlistInqVarZaxis(vlistID, varID);
+      int zaxisID = vlistInqVarZaxis(vlistID, recinfo->varID);
       int zaxisType = zaxisInqType(zaxisID);
       CdiDateTime vDateTime = streamptr1->tsteps[tsID].taxis.vDateTime;
 
@@ -443,7 +443,7 @@ grb_write_var_slice(stream_t *streamptr, int varID, int levelID, int memtype, co
     case CDI_PROTOCOL_FILE:
       {
         size_t (*myFileWrite)(int fileID, const void *restrict buffer, size_t len)
-            = (size_t(*)(int, const void *restrict, size_t)) namespaceSwitchGet(NSSWITCH_FILE_WRITE).func;
+            = (size_t (*)(int, const void *restrict, size_t)) namespaceSwitchGet(NSSWITCH_FILE_WRITE).func;
 
         size_t nwrite = myFileWrite(fileID, gribbuffer, nbytes);
         if (nwrite != nbytes) SysError("Failed to write GRIB slice!");
diff --git a/src/gribapi.h b/src/gribapi.h
index 87e9a23e30fa062572fc3314a9e18d5a71d8fdf7..784eb940a8e64e75d1f69e54d2b8fa50645ed675 100644
--- a/src/gribapi.h
+++ b/src/gribapi.h
@@ -49,6 +49,7 @@
 #define  GRIB2_GTYPE_LATLON_ROTSTR         3  // Stretched and Rotated Latitude/longitude
 #define  GRIB2_GTYPE_STERE                20  // Polar stereographic projection
 #define  GRIB2_GTYPE_LCC                  30  // Lambert conformal
+#define  GRIB2_GTYPE_LLAM                 33  // Lambert LAM
 #define  GRIB2_GTYPE_GAUSSIAN             40  // Gaussian latitude/longitude
 #define  GRIB2_GTYPE_GAUSSIAN_ROT         41  // Rotated Gaussian latitude/longitude
 #define  GRIB2_GTYPE_GAUSSIAN_STR         42  // Stretched Gaussian latitude/longitude
diff --git a/src/gribapi_utilities.c b/src/gribapi_utilities.c
index 7d45150803ce435a1b184442906c4a5011589fcc..ee5b158633b7760dac7ff66825c1b92c778cda80 100644
--- a/src/gribapi_utilities.c
+++ b/src/gribapi_utilities.c
@@ -591,6 +591,7 @@ gribapiGetGridType(grib_handle *gh)
     case GRIB2_GTYPE_GAUSSIAN: return has_ni(gh) ? GRID_GAUSSIAN : GRID_GAUSSIAN_REDUCED;
     case GRIB2_GTYPE_LATLON_ROT: return GRID_PROJECTION;
     case GRIB2_GTYPE_LCC: return CDI_PROJ_LCC;
+    case GRIB2_GTYPE_LLAM: return CDI_PROJ_LCC;  // Handle LLAM as LCC
     case GRIB2_GTYPE_STERE: return CDI_PROJ_STERE;
     case GRIB2_GTYPE_SPECTRAL: return GRID_SPECTRAL;
     case GRIB2_GTYPE_GME: return GRID_GME;
@@ -966,7 +967,7 @@ gribapiGetGrid(grib_handle *gh, grid_t *grid)
          Default / implicit scanning mode is 64:
                             i and j scan positively, i points are consecutive (row-major)        */
 #ifdef HIRLAM_EXTENSIONS
-      if (cdiDebugExt >= 30 && edititionnumber <= 1)
+      if (cdiDebugExt >= 30 && editionNumber <= 1)
         {
           //  indicatorOfParameter=33,indicatorOfTypeOfLevel=105,level
           long paramId, levelTypeId, levelId;
diff --git a/src/grid.c b/src/grid.c
index 9e2897e9209f7d10922920ebd88fa2ae56c1ed0b..f29d333f0c7b76978ecb6972a1527d3c3db72380 100644
--- a/src/grid.c
+++ b/src/grid.c
@@ -989,9 +989,7 @@ gridInqSize(int gridID)
     {
       size_t xsize = gridptr->x.size;
       size_t ysize = gridptr->y.size;
-
       size = ysize ? xsize * ysize : xsize;
-
       gridptr->size = size;
     }
 
@@ -2194,7 +2192,7 @@ compareXYvals2(grid_t *gridRef, grid_t *gridTest)
                 || ((gridTest->x.bounds == NULL) ^ (gridRef->x.bounds == NULL))
                 || ((gridTest->y.bounds == NULL) ^ (gridRef->y.bounds == NULL));
 
-  typedef double (*inqVal)(grid_t * grid, SizeType index);
+  typedef double (*inqVal)(grid_t *grid, SizeType index);
   inqVal inqXValRef = gridRef->vtable->inqXVal, inqYValRef = gridRef->vtable->inqYVal, inqXValTest = gridTest->vtable->inqXVal,
          inqYValTest = gridTest->vtable->inqYVal;
 
@@ -4871,7 +4869,7 @@ cdiVlistAddGridIfNew(int vlistID, grid_t *grid, int mode)
         }
     }
 
-  return (struct addIfNewRes){ .Id = gridID, .isNew = (!gridIsDefined && !gridIsDefinedGlobal) };
+  return (struct addIfNewRes) { .Id = gridID, .isNew = (!gridIsDefined && !gridIsDefinedGlobal) };
 }
 
 const struct gridVirtTable cdiGridVtable = {
diff --git a/src/ieg.h b/src/ieg.h
index df110c2d5edc2c34853efe6591cfe66a12aa38ca..cb07281d67a51e8841c0403cfb28bacb8cf57569 100644
--- a/src/ieg.h
+++ b/src/ieg.h
@@ -1,6 +1,8 @@
 #ifndef _IEG_H
 #define _IEG_H
 
+#include <stdlib.h>
+
 // clang-format off
 
 /* Level Types */
diff --git a/src/ieglib.c b/src/ieglib.c
index 067ce68abc22b6c4b118febf91cde188a44d27dd..b5e08e2cf245c7638efa8cb48f0cc6267a91214e 100644
--- a/src/ieglib.c
+++ b/src/ieglib.c
@@ -11,7 +11,6 @@
  * next to find out what happened */
 
 #include <stdio.h>
-#include <stdlib.h>
 #include <string.h>
 #include <ctype.h>
 
diff --git a/src/input_file.c b/src/input_file.c
index 40dd475037c44dc306844bb311a8c05327823216..11df000c99bb693605e2d3694b529d8b99287d14 100644
--- a/src/input_file.c
+++ b/src/input_file.c
@@ -18,9 +18,20 @@
 
 #include <errno.h>
 #include <fcntl.h>
-#include <pthread.h>
 #include <string.h>
+
+// On Windows, define ssize_t and pread manually
+#ifdef _WIN32
+#define ssize_t __int64
+#define pread read
+#include <io.h>
+#else
 #include <unistd.h>
+#endif
+
+#if HAVE_PTHREAD
+#include <pthread.h>
+#endif
 
 static void cdiInputFile_destruct(CdiInputFile *me);
 
@@ -64,7 +75,9 @@ success:
 
 static CdiInputFile **openFileList = NULL;
 static size_t openFileCount = 0, openFileListSize = 0;
+#if HAVE_PTHREAD
 static pthread_mutex_t openFileListLock = PTHREAD_MUTEX_INITIALIZER;
+#endif
 
 // This either returns a new object, or retains and returns a preexisting open file.
 CdiInputFile *
@@ -72,8 +85,10 @@ cdiInputFile_make(const char *path)
 {
   CdiInputFile *result = NULL;
   xassert(path);
+#if HAVE_PTHREAD
   int error = pthread_mutex_lock(&openFileListLock);
   xassert(!error);
+#endif
   {
     // Check the list of open files for the given path.
     for (size_t i = openFileCount; i-- && !result;)
@@ -108,8 +123,10 @@ cdiInputFile_make(const char *path)
           }
       }
   }
+#if HAVE_PTHREAD
   error = pthread_mutex_unlock(&openFileListLock);
   xassert(!error);
+#endif
   return result;
 }
 
@@ -142,8 +159,10 @@ cdiInputFile_getPath(const CdiInputFile *me)
 void
 cdiInputFile_destruct(CdiInputFile *me)
 {
+#if HAVE_PTHREAD
   int error = pthread_mutex_lock(&openFileListLock);
   xassert(!error);
+#endif
   {
     // Find the position of me in the list of open files.
     ssize_t position = (ssize_t) openFileCount;
@@ -152,8 +171,10 @@ cdiInputFile_destruct(CdiInputFile *me)
     // Remove me from the list
     openFileList[position] = openFileList[--openFileCount];
   }
+#if HAVE_PTHREAD
   error = pthread_mutex_unlock(&openFileListLock);
   xassert(!error);
+#endif
   cdiInputFile_condestruct(me, NULL);
 }
 
diff --git a/src/institution.c b/src/institution.c
index 0250a7dc7045fbe51bbec18d9d3a7eba1a428bfa..6484ddb0601aad0ad1065975ef8bcb2139e8c973 100644
--- a/src/institution.c
+++ b/src/institution.c
@@ -15,7 +15,6 @@
 #include <limits.h>
 
 #include "cdi.h"
-#include "cdi_int.h"
 #include "resource_handle.h"
 #include "resource_unpack.h"
 #include "namespace.h"
diff --git a/src/iterator_fallback.c b/src/iterator_fallback.c
index eb561f4d11716a67f6e37fa39993e2f2a4962361..fd7bc6db7bc393b64ff4c1be216159d30d0abed8 100644
--- a/src/iterator_fallback.c
+++ b/src/iterator_fallback.c
@@ -20,6 +20,13 @@
 #include <limits.h>
 #include <stdlib.h>
 
+// On Windows, define ssize_t manually
+#ifdef _WIN32
+#define ssize_t __int64
+#else
+#include <unistd.h>
+#endif
+
 struct CdiFallbackIterator
 {
   CdiIterator super;
diff --git a/src/make_fint.c b/src/make_fint.c
index 6d76942d3d6c3671334c5aeb5ca4fc36f37717f9..efc2165972f376d5d1f35bc8200c3db9ac70fbf4 100644
--- a/src/make_fint.c
+++ b/src/make_fint.c
@@ -1367,8 +1367,8 @@ static int detectComment(char **line_, ssize_t *lineLen, size_t *lineBufSize,
             cline += strspn(cline, " \t*");
             char *eol = strchr(cline, '\n');
             if (!eol) eol = comment + commentLen;
-            size_t lineLen = (size_t)(eol - cline);
-            fprintf(fpinc, "!  %.*s\n", (int)lineLen, cline);
+            size_t lineLen2 = (size_t)(eol - cline);
+            fprintf(fpinc, "!  %.*s\n", (int)lineLen2, cline);
             cline = (eol != comment + commentLen) ? eol + 1: NULL;
           } while (cline);
           fputs("!\n", fpinc);
diff --git a/src/pio_server.c b/src/pio_server.c
index bdda03d49c0726350b4daae0ffbfecbd6ceb01b5..a9f4c7c3dd66ef03bcd985ccaae950e450448bbc 100644
--- a/src/pio_server.c
+++ b/src/pio_server.c
@@ -246,8 +246,8 @@ readFuncCall(struct winHeaderEntry *header, size_t streamIdx)
         int position = header->offset;
         int changedTaxisID = taxisUnpack((char *) rxWin[streamIdx].clientBuf[0].mem, (int) rxWin[streamIdx].clientBuf[0].size,
                                          &position, originNamespace, &pioInterComm, 0);
-        taxis_t *oldTaxisPtr = taxisPtr(oldTaxisID);
-        taxis_t *changedTaxisPtr = taxisPtr(changedTaxisID);
+        taxis_t *oldTaxisPtr = taxis_to_pointer(oldTaxisID);
+        taxis_t *changedTaxisPtr = taxis_to_pointer(changedTaxisID);
         ptaxisCopy(oldTaxisPtr, changedTaxisPtr);
         taxisDestroy(changedTaxisID);
         streamDefTimestep(streamID, funcArgs->streamNewTimestep.tsID);
diff --git a/src/serialize.h b/src/serialize.h
index d771d704fcddae3935dae49c9361a7250b4a3761..c77c56707ac251c7867a1991f9a618f27a47ab38 100644
--- a/src/serialize.h
+++ b/src/serialize.h
@@ -42,16 +42,16 @@ serializeKeysGetPackSize(const cdi_keys_t *keysp, void *context)
       int type = keyp->type;
       packBuffSize += serializeGetSize(1, CDI_DATATYPE_INT, context);  // key
       packBuffSize += serializeGetSize(1, CDI_DATATYPE_INT, context);  // type
-      if (type == KEY_BYTES)
+      if (type == KeyBytes)
         {
           int length = keyp->length;
           packBuffSize += serializeGetSize(1, CDI_DATATYPE_INT, context) + serializeGetSize(length, CDI_DATATYPE_TXT, context);
         }
-      else if (type == KEY_INT)
+      else if (type == KeyInt)
         {
           packBuffSize += serializeGetSize(1, CDI_DATATYPE_INT, context);
         }
-      else if (type == KEY_FLOAT)
+      else if (type == KeyFloat)
         {
           packBuffSize += serializeGetSize(1, CDI_DATATYPE_FLT64, context);
         }
@@ -74,18 +74,18 @@ serializeKeysPack(const cdi_keys_t *keysp, void *buf, int buf_size, int *positio
       int type = keyp->type;
       serializePack(&key, 1, CDI_DATATYPE_INT, buf, buf_size, position, context);
       serializePack(&type, 1, CDI_DATATYPE_INT, buf, buf_size, position, context);
-      if (type == KEY_BYTES)
+      if (type == KeyBytes)
         {
           int length = keyp->length;
           serializePack(&length, 1, CDI_DATATYPE_INT, buf, buf_size, position, context);
           serializePack(keyp->v.s, length, CDI_DATATYPE_TXT, buf, buf_size, position, context);
           d ^= cdiCheckSum(CDI_DATATYPE_TXT, length, keyp->v.s);
         }
-      else if (type == KEY_INT)
+      else if (type == KeyInt)
         {
           serializePack(&keyp->v.i, 1, CDI_DATATYPE_INT, buf, buf_size, position, context);
         }
-      else if (type == KEY_FLOAT)
+      else if (type == KeyFloat)
         {
           serializePack(&keyp->v.d, 1, CDI_DATATYPE_FLT64, buf, buf_size, position, context);
         }
@@ -108,7 +108,7 @@ serializeKeysUnpack(const void *buf, int buf_size, int *position, cdi_keys_t *ke
       int key, type;
       serializeUnpack(buf, buf_size, position, &key, 1, CDI_DATATYPE_INT, context);
       serializeUnpack(buf, buf_size, position, &type, 1, CDI_DATATYPE_INT, context);
-      if (type == KEY_BYTES)
+      if (type == KeyBytes)
         {
           int length;
           serializeUnpack(buf, buf_size, position, &length, 1, CDI_DATATYPE_INT, context);
@@ -121,13 +121,13 @@ serializeKeysUnpack(const void *buf, int buf_size, int *position, cdi_keys_t *ke
           cdiDefVarKeyBytes(keysp, key, (unsigned char *) buffer, length);
           d2 ^= cdiCheckSum(CDI_DATATYPE_TXT, length, buffer);
         }
-      else if (type == KEY_INT)
+      else if (type == KeyInt)
         {
           int ival;
           serializeUnpack(buf, buf_size, position, &ival, 1, CDI_DATATYPE_INT, context);
           cdiDefVarKeyInt(keysp, key, ival);
         }
-      else if (type == KEY_FLOAT)
+      else if (type == KeyFloat)
         {
           double dval;
           serializeUnpack(buf, buf_size, position, &dval, 1, CDI_DATATYPE_FLT64, context);
diff --git a/src/service.h b/src/service.h
index 9c9a5d14b39cd85e57340cd2d8f94170258d253f..edc06842ea2e1fd3de959f67cb4fb73aa5fa6499 100644
--- a/src/service.h
+++ b/src/service.h
@@ -1,6 +1,8 @@
 #ifndef _SERVICE_H
 #define _SERVICE_H
 
+#include <stdlib.h>
+
 typedef struct
 {
   int checked;
diff --git a/src/servicelib.c b/src/servicelib.c
index 70a649f968393f12fdf25cd8c0fd206e4fdee784..5c0bd113caf78434894f0b6e75d14610236cc32c 100644
--- a/src/servicelib.c
+++ b/src/servicelib.c
@@ -11,7 +11,6 @@
  * next to find out what happened */
 
 #include <stdio.h>
-#include <stdlib.h>
 #include <stdarg.h>
 #include <string.h>
 #include <ctype.h>
diff --git a/src/stream.c b/src/stream.c
index 81fee3cae624cb8f00ecfc238bb57ca3c65d0dcb..f3a11f7c81df0494610670f4a9aa27cea5621718 100644
--- a/src/stream.c
+++ b/src/stream.c
@@ -251,9 +251,10 @@ streamDefNumWorker(int streamID, int numWorker)
               if (streamptr->maxGlobalRecs == -1) xabort("Internal error: number of timesteps missing!");
               if (streamptr->maxGlobalRecs == 1) numWorker = 0;
               if (numWorker > streamptr->maxGlobalRecs) numWorker = (int) streamptr->maxGlobalRecs;
-              if (streamptr->chunkSizeTdim > 1 && numWorker > streamptr->nvars) numWorker = streamptr->nvars;
-              if (streamptr->chunkSizeZdim > 1) numWorker = 0;
-              if (CDI_Test) Message("chunkSizeTdim=%d chunkSizeZdim=%d", streamptr->chunkSizeTdim, streamptr->chunkSizeZdim);
+              CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+              if (cdfInfo->chunkSizeDimT > 1 && numWorker > streamptr->nvars) numWorker = streamptr->nvars;
+              if (cdfInfo->chunkSizeDimZ > 1) numWorker = 0;
+              if (CDI_Test) Message("chunkSizeTdim=%d chunkSizeZdim=%d", cdfInfo->chunkSizeDimT, cdfInfo->chunkSizeDimZ);
             }
 #endif
           else
@@ -429,7 +430,7 @@ cdiInqContents(stream_t *streamptr)
       if (taxisID != CDI_UNDEFID)
         {
           taxis_t *taxisptr1 = &streamptr->tsteps[0].taxis;
-          taxis_t *taxisptr2 = taxisPtr(taxisID);
+          taxis_t *taxisptr2 = taxis_to_pointer(taxisID);
           ptaxisCopy(taxisptr2, taxisptr1);
         }
     }
@@ -1076,31 +1077,32 @@ streamDefaultValue(stream_t *streamptr)
   basetimeInit(&streamptr->basetime);
 
 #ifdef HAVE_LIBNETCDF
-  streamptr->nc_complex_float_id = CDI_UNDEFID;
-  streamptr->nc_complex_double_id = CDI_UNDEFID;
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  cdfInfo->complexFloatId = CDI_UNDEFID;
+  cdfInfo->complexDoubleId = CDI_UNDEFID;
 
-  for (int i = 0; i < MAX_ZAXES_PS; i++) streamptr->zaxisID[i] = CDI_UNDEFID;
-  for (int i = 0; i < MAX_ZAXES_PS; i++) streamptr->nczvarID[i] = CDI_UNDEFID;
+  for (int i = 0; i < MAX_ZAXES_PS; i++) cdfInfo->zaxisIdVec[i] = CDI_UNDEFID;
+  for (int i = 0; i < MAX_ZAXES_PS; i++) cdfInfo->ncZvarIdVec[i] = CDI_UNDEFID;
 
   for (int i = 0; i < MAX_GRIDS_PS; i++)
     {
-      streamptr->ncgrid[i].start = CDI_UNDEFID;
-      streamptr->ncgrid[i].count = CDI_UNDEFID;
-      streamptr->ncgrid[i].gridID = CDI_UNDEFID;
-      for (size_t j = 0; j < CDF_SIZE_ncIDs; ++j) streamptr->ncgrid[i].ncIDs[j] = CDI_UNDEFID;
+      cdfInfo->cdfGridVec[i].start = CDI_UNDEFID;
+      cdfInfo->cdfGridVec[i].count = CDI_UNDEFID;
+      cdfInfo->cdfGridVec[i].gridID = CDI_UNDEFID;
+      for (size_t j = 0; j < CDF_SIZE_NCID; ++j) cdfInfo->cdfGridVec[i].ncIdVec[j] = CDI_UNDEFID;
     }
 
-  streamptr->ncNumDims = 0;
-  for (int i = 0; i < MAX_DIMS_PS; i++) streamptr->ncDimID[i] = CDI_UNDEFID;
-  for (int i = 0; i < MAX_DIMS_PS; i++) streamptr->ncDimLen[i] = 0;
+  cdfInfo->ncNumDims = 0;
+  for (int i = 0; i < MAX_DIMS_PS; i++) cdfInfo->ncDimIdVec[i] = CDI_UNDEFID;
+  for (int i = 0; i < MAX_DIMS_PS; i++) cdfInfo->ncDimLenVec[i] = 0;
 
-  streamptr->vct.ilev = 0;
-  streamptr->vct.mlev = 0;
-  streamptr->vct.ilevID = CDI_UNDEFID;
-  streamptr->vct.mlevID = CDI_UNDEFID;
+  cdfInfo->chunkSizeDimT = 0;
+  cdfInfo->chunkSizeDimZ = 0;
 
-  streamptr->chunkSizeTdim = 0;
-  streamptr->chunkSizeZdim = 0;
+  cdfInfo->vct.ilev = 0;
+  cdfInfo->vct.mlev = 0;
+  cdfInfo->vct.ilevID = CDI_UNDEFID;
+  cdfInfo->vct.mlevID = CDI_UNDEFID;
 #endif
   streamptr->maxGlobalRecs = CDI_UNDEFID;
 
@@ -1233,16 +1235,12 @@ cdiStreamCloseDefaultDelegate(stream_t *streamptr, int recordBufIsToBeDeleted)
         cdfClose(fileID);
         if (streamptr->ntsteps == 0 && streamptr->tsteps != NULL)
           {
-            if (streamptr->tsteps[0].records)
-              {
-                Free(streamptr->tsteps[0].records);
-                streamptr->tsteps[0].records = NULL;
-              }
-            if (streamptr->tsteps[0].recIDs)
-              {
-                Free(streamptr->tsteps[0].recIDs);
-                streamptr->tsteps[0].recIDs = NULL;
-              }
+            tsteps_t *tstep = &(streamptr->tsteps[0]);
+            // clang-format off
+            if (tstep->recinfo) { Free(tstep->recinfo); tstep->recinfo = NULL; }
+            if (tstep->records) { Free(tstep->records); tstep->records = NULL; }
+            if (tstep->recIDs) { Free(tstep->recIDs); tstep->recIDs = NULL; }
+            // clang-format on
           }
         break;
       }
@@ -1306,8 +1304,11 @@ streamDestroyViaDelegate(stream_t *streamptr, void (*streamCloseDelegate)(stream
       for (int index = 0; index < maxSteps; ++index)
         {
           tsteps_t *tstep = &(streamptr->tsteps[index]);
-          if (tstep->records) Free(tstep->records);
-          if (tstep->recIDs) Free(tstep->recIDs);
+          // clang-format off
+          if (tstep->recinfo) { Free(tstep->recinfo); tstep->recinfo = NULL; }
+          if (tstep->records) { Free(tstep->records); tstep->records = NULL; }
+          if (tstep->recIDs) { Free(tstep->recIDs); tstep->recIDs = NULL; }
+          // clang-format on
           taxisDestroyKernel(&(tstep->taxis));
         }
 
@@ -1339,7 +1340,6 @@ streamDestroyViaDelegate(stream_t *streamptr, void (*streamCloseDelegate)(stream
   Free(streamptr);
 }
 
-
 static void
 streamDestroy(stream_t *streamptr)
 {
@@ -1462,14 +1462,15 @@ cdiStreamDefTimestep_(stream_t *streamptr, int tsID)
     }
 
   int taxisID = vlistInqTaxis(vlistID);
-  if (taxisID != CDI_UNDEFID) ptaxisCopy(&streamptr->tsteps[tsID].taxis, taxisPtr(taxisID));
+  if (taxisID != CDI_UNDEFID) ptaxisCopy(&streamptr->tsteps[tsID].taxis, taxis_to_pointer(taxisID));
 
   streamptr->curTsID = tsID;
   streamptr->ntsteps = tsID + 1;
 
+  bool baseFiletypeIsNetCDF = (cdiBaseFiletype(streamptr->filetype) == CDI_FILETYPE_NETCDF);
 #ifdef HAVE_LIBNETCDF
   int timeIsVarying = vlistHasTime(vlistID);
-  if (cdiBaseFiletype(streamptr->filetype) == CDI_FILETYPE_NETCDF && timeIsVarying)
+  if (baseFiletypeIsNetCDF && timeIsVarying)
     {
       /* usually points to cdfDefTimestep in serial mode but
        * to cdiPioCdfDefTimestep on servers and to a null-op on
@@ -1480,7 +1481,7 @@ cdiStreamDefTimestep_(stream_t *streamptr, int tsID)
     }
 #endif
 
-  cdi_create_records(streamptr, tsID);
+  cdi_create_records(streamptr, tsID, !baseFiletypeIsNetCDF);
 
   return (int) streamptr->ntsteps;
 }
@@ -1563,7 +1564,7 @@ streamInqTimestep(int streamID, int tsID)
       streamptr->tsteps[tsID].curRecID = CDI_UNDEFID;
       int taxisID = vlistInqTaxis(vlistID);
       if (taxisID == -1) Error("Timestep undefined for fileID = %d", streamID);
-      ptaxisCopy(taxisPtr(taxisID), &streamptr->tsteps[tsID].taxis);
+      ptaxisCopy(taxis_to_pointer(taxisID), &streamptr->tsteps[tsID].taxis);
 
       return nrecs;
     }
@@ -1632,7 +1633,7 @@ streamInqTimestep(int streamID, int tsID)
   int taxisID = vlistInqTaxis(vlistID);
   if (taxisID == -1) Error("Timestep undefined for fileID = %d", streamID);
 
-  ptaxisCopy(taxisPtr(taxisID), &streamptr->tsteps[tsID].taxis);
+  ptaxisCopy(taxis_to_pointer(taxisID), &streamptr->tsteps[tsID].taxis);
 
   return nrecs;
 }
@@ -1783,26 +1784,6 @@ streamInqFileID(int streamID)
   return s->fileID;
 }
 
-void
-cdiDefAccesstype(int streamID, int type)
-{
-  stream_t *s = (stream_t *) reshGetVal(streamID, &streamOps);
-
-  if (s->accesstype == CDI_UNDEFID)
-    {
-      s->accesstype = type;
-    }
-  else if (s->accesstype != type)
-    Error("Changing access type from %s not allowed!", s->accesstype == TYPE_REC ? "REC to VAR" : "VAR to REC");
-}
-
-int
-cdiInqAccesstype(int streamID)
-{
-  stream_t *s = (stream_t *) reshGetVal(streamID, &streamOps);
-  return s->accesstype;
-}
-
 static int
 streamTxCode(void *s)
 {
@@ -1851,7 +1832,7 @@ cdiStreamSetupVlist_(stream_t *streamptr, int vlistID)
           if (taxisInqType(taxisID) == TAXIS_RELATIVE)
             if (cdiBaseFiletype(streamptr->filetype) == CDI_FILETYPE_NETCDF)
               {
-                const taxis_t *taxisptr = taxisPtr(taxisID);
+                const taxis_t *taxisptr = taxis_to_pointer(taxisID);
                 if (cdiDateTime_isNull(taxisptr->rDateTime))
                   {
                     int vdate = taxisInqVdate(taxisID);
@@ -1860,7 +1841,7 @@ cdiStreamSetupVlist_(stream_t *streamptr, int vlistID)
                   }
               }
 #endif
-          ptaxisCopy(&streamptr->tsteps[0].taxis, taxisPtr(taxisID));
+          ptaxisCopy(&streamptr->tsteps[0].taxis, taxis_to_pointer(taxisID));
         }
 
       switch (cdiBaseFiletype(streamptr->filetype))
@@ -2039,13 +2020,13 @@ streamUnpack(char *unpackBuffer, int unpackBufferSize, int *unpackBufferPos, int
   return retval;
 }
 
-
 /* *
  * This function does not really close the memio,
  * this has to be done outside cdi to access the memory buffer*/
 void
 freePtrAfterNCMem(stream_t *streamptr, int recordBufIsToBeDeleted)
 {
+  (void) recordBufIsToBeDeleted;
   int fileID = streamptr->fileID;
 
   if (fileID == CDI_UNDEFID)
@@ -2056,20 +2037,15 @@ freePtrAfterNCMem(stream_t *streamptr, int recordBufIsToBeDeleted)
 
   if (streamptr->ntsteps == 0 && streamptr->tsteps != NULL)
     {
-      if (streamptr->tsteps[0].records)
-        {
-          Free(streamptr->tsteps[0].records);
-          streamptr->tsteps[0].records = NULL;
-        }
-      if (streamptr->tsteps[0].recIDs)
-        {
-          Free(streamptr->tsteps[0].recIDs);
-          streamptr->tsteps[0].recIDs = NULL;
-        }
+      tsteps_t *tstep = &(streamptr->tsteps[0]);
+      // clang-format off
+      if (tstep->recinfo) { Free(tstep->recinfo); tstep->recinfo = NULL; }
+      if (tstep->records) { Free(tstep->records); tstep->records = NULL; }
+      if (tstep->recIDs) { Free(tstep->recIDs); tstep->recIDs = NULL; }
+      // clang-format on
     }
 }
 
-
 void
 streamCloseNCMem(int streamID)
 {
@@ -2079,7 +2055,7 @@ streamCloseNCMem(int streamID)
   if (lockIO) CDI_IO_LOCK();
 
   if (CDI_Debug) Message("streamID = %d filename = %s", streamID, streamptr->filename);
-  streamDestroyViaDelegate(streamptr,freePtrAfterNCMem);
+  streamDestroyViaDelegate(streamptr, freePtrAfterNCMem);
   reshRemove(streamID, &streamOps);
   if (CDI_Debug) Message("Removed stream %d from stream list", streamID);
 
diff --git a/src/stream_cdf.h b/src/stream_cdf.h
index d4b379742b09f9983910c0310f8c0220e3eac2e8..8d61ec280aed2c29eb03cb1f523adb3301d0adbf 100644
--- a/src/stream_cdf.h
+++ b/src/stream_cdf.h
@@ -32,7 +32,7 @@ int cdfInqContents(stream_t *streamptr);
 void cdfEndDef(stream_t *streamptr);
 void cdfDefField(stream_t *streamptr);
 
-void cdfCopyRecord(stream_t *streamptr2, stream_t *streamptr1);
+void cdfCopyField(stream_t *streamptr2, stream_t *streamptr1);
 
 void cdfDefineAttributes(int filetype, int vlistID, int varID, int fileID, int ncvarID);
 
diff --git a/src/stream_cdf_i.c b/src/stream_cdf_i.c
index 9b7565a012bdd95b2b93a32f63d07e6db6648768..3096c518c8518a88ae9114558489334cfc903126 100644
--- a/src/stream_cdf_i.c
+++ b/src/stream_cdf_i.c
@@ -6,7 +6,6 @@
 
 #include <ctype.h>
 #include <limits.h>
-#include <strings.h>
 
 #include "dmemory.h"
 #include "cdi_int.h"
@@ -19,6 +18,13 @@
 #include "cdf_lazy_grid.h"
 #include "cdf_filter.h"
 
+// On Windows, define strcasecmp manually
+#ifdef _WIN32
+#define strcasecmp _stricmp
+#else
+#include <unistd.h>
+#endif
+
 enum VarStatus
 {
   UndefVar = -1,
@@ -52,6 +58,7 @@ typedef struct
 
 typedef struct
 {
+  int cdiVarID;
   int ncid;
   int varStatus;
   bool ignoreVar;
@@ -115,6 +122,7 @@ typedef struct
   size_t gridSize;
   size_t xSize;
   size_t ySize;
+  size_t zSize;
   int nattsNC;
   int natts;
   int *atts;
@@ -138,7 +146,6 @@ typedef struct
   char longname[CDI_MAX_NAME];
   char stdname[CDI_MAX_NAME];
   char units[CDI_MAX_NAME];
-  char extra[CDI_MAX_NAME];
   char filterSpec[CDI_MAX_NAME];
 } ncvar_t;
 
@@ -343,7 +350,8 @@ cdfInqDatatype(stream_t *streamptr, int xtype, bool isUnsigned)
   else if (xtype == NC_UINT64)  datatype = CDI_DATATYPE_FLT64;
   else
     {
-      if (xtype != streamptr->nc_complex_float_id && xtype != streamptr->nc_complex_double_id)
+      CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+      if (xtype != cdfInfo->complexFloatId && xtype != cdfInfo->complexDoubleId)
         {
           bool isUserDefinedType = false;
 #ifdef NC_FIRSTUSERTYPEID
@@ -362,14 +370,14 @@ cdfInqDatatype(stream_t *streamptr, int xtype, bool isUnsigned)
                   nc_inq_compound_field(fileID, xtype, 1, NULL, NULL, &field_type2, &field_dims2, NULL);
                   if (field_type1 == field_type2 && field_dims1 == 0 && field_dims2 == 0)
                     {
-                      if      (field_type1 == NC_FLOAT)  streamptr->nc_complex_float_id = xtype;
-                      else if (field_type1 == NC_DOUBLE) streamptr->nc_complex_double_id = xtype;
+                      if      (field_type1 == NC_FLOAT)  cdfInfo->complexFloatId = xtype;
+                      else if (field_type1 == NC_DOUBLE) cdfInfo->complexDoubleId = xtype;
                     }
                 }
             }
         }
-      if      (xtype == streamptr->nc_complex_float_id )  datatype = CDI_DATATYPE_CPX32;
-      else if (xtype == streamptr->nc_complex_double_id)  datatype = CDI_DATATYPE_CPX64;
+      if      (xtype == cdfInfo->complexFloatId )  datatype = CDI_DATATYPE_CPX32;
+      else if (xtype == cdfInfo->complexDoubleId)  datatype = CDI_DATATYPE_CPX64;
     }
 #endif
   // clang-format on
@@ -549,7 +557,7 @@ cdf_time_dimid(int fileID, int ndims, ncdim_t *ncdims, int nvars, ncvar_t *ncvar
               check_dimids[dimid0] = true;
 
               char sbuf[CDI_MAX_NAME];
-              for (int iatt = 0; iatt < ncvar->nattsNC; ++iatt)
+              for (int iatt = 0, n = ncvar->nattsNC; iatt < n; ++iatt)
                 {
                   sbuf[0] = 0;
                   cdf_inq_attname(fileID, varid, iatt, sbuf);
@@ -586,6 +594,7 @@ init_ncvars(int nvars, ncvar_t *ncvars, int ncid)
   for (int varid = 0; varid < nvars; varid++)
     {
       ncvar_t *ncvar = &ncvars[varid];
+      ncvar->cdiVarID = CDI_UNDEFID;
       ncvar->ncid = ncid;
       ncvar->varStatus = UndefVar;
       ncvar->ignoreVar = false;
@@ -649,6 +658,7 @@ init_ncvars(int nvars, ncvar_t *ncvars, int ncid)
       ncvar->gridSize = 0;
       ncvar->xSize = 0;
       ncvar->ySize = 0;
+      ncvar->zSize = 0;
       ncvar->nattsNC = 0;
       ncvar->natts = 0;
       ncvar->atts = NULL;
@@ -673,7 +683,6 @@ init_ncvars(int nvars, ncvar_t *ncvars, int ncid)
       memset(ncvar->longname, 0, CDI_MAX_NAME);
       memset(ncvar->stdname, 0, CDI_MAX_NAME);
       memset(ncvar->units, 0, CDI_MAX_NAME);
-      memset(ncvar->extra, 0, CDI_MAX_NAME);
       memset(ncvar->filterSpec, 0, CDI_MAX_NAME);
     }
 }
@@ -724,14 +733,14 @@ scan_hybrid_formulaterms(int ncid, int ncfvarid, int *avarid, int *bvarid, int *
       char *tagname = pstring;
       while (!isspace((int) *pstring) && *pstring != 0) pstring++;
       if (*pstring == 0) lstop = true;
-      *pstring++ = 0;
+      *(pstring++) = 0;
 
       while (isspace((int) *pstring)) pstring++;
       if (*pstring == 0) break;
       char *varname = pstring;
       while (!isspace((int) *pstring) && *pstring != 0) pstring++;
       if (*pstring == 0) lstop = true;
-      *pstring++ = 0;
+      *(pstring++) = 0;
 
       int dimvarid;
       int status_nc = nc_inq_varid(ncid, varname, &dimvarid);
@@ -1031,11 +1040,11 @@ cdf_get_cell_varid(char *attstring, int ncid)
   while (isspace((int) *pstring)) pstring++;
   char *cell_measures = pstring;
   while (isalnum((int) *pstring)) pstring++;
-  *pstring++ = 0;
+  *(pstring++) = 0;
   while (isspace((int) *pstring)) pstring++;
   char *cell_var = pstring;
   while (!isspace((int) *pstring) && *pstring != 0) pstring++;
-  *pstring++ = 0;
+  *(pstring++) = 0;
   /*
     printf("cell_measures >%s<\n", cell_measures);
     printf("cell_var >%s<\n", cell_var);
@@ -1054,21 +1063,6 @@ cdf_get_cell_varid(char *attstring, int ncid)
   return nc_cell_id;
 }
 
-static void
-set_extra_attr(char *buf, int nvdims, const size_t *chunks)
-{
-  size_t pos = strlen(buf);
-  static const char prefix[] = "chunks=";
-  memcpy(buf + pos, prefix, sizeof(prefix));
-  pos += sizeof(prefix) - 1;
-  for (int i = nvdims - 1; i >= 0; --i)
-    {
-      pos += (size_t) (snprintf(buf + pos, CDI_MAX_NAME - pos, "%zu%s", chunks[i], i > 0 ? "x" : ""));
-    }
-  buf[pos] = ' ';
-  buf[pos + 1] = 0;
-}
-
 static bool
 is_valid_coordinate(ncvar_t *ncvar)
 {
@@ -1090,7 +1084,7 @@ read_coordinates_vars(int ncid, char *attstring, ncvar_t *ncvar, ncvar_t *ncvars
       while (!isspace((int) *attstring) && *attstring != 0) attstring++;
       if (*attstring == 0) lstop = true;
       if (*(attstring - 1) == ',') *(attstring - 1) = 0;
-      *attstring++ = 0;
+      *(attstring++) = 0;
 
       int dimvarid;
       int status = nc_inq_varid(ncid, varname, &dimvarid);
@@ -1116,7 +1110,7 @@ read_coordinates_vars(int ncid, char *attstring, ncvar_t *ncvar, ncvar_t *ncvars
 
           if (k == *nchecked_vars)
             {
-              if (*nchecked_vars < max_check_vars) checked_vars[*nchecked_vars++] = strdup(varname);
+              if (*nchecked_vars < max_check_vars) checked_vars[(*nchecked_vars)++] = strdup(varname);
               Warning("%s - >%s<", nc_strerror(status), varname);
             }
         }
@@ -1134,7 +1128,7 @@ read_auxiliary_vars(int ncid, char *attstring, ncvar_t *ncvar, ncvar_t *ncvars)
       char *varname = attstring;
       while (!isspace((int) *attstring) && *attstring != 0) attstring++;
       if (*attstring == 0) lstop = true;
-      *attstring++ = 0;
+      *(attstring++) = 0;
 
       int dimvarid;
       int status = nc_inq_varid(ncid, varname, &dimvarid);
@@ -1167,7 +1161,7 @@ set_vars_chunks(int ncid, int ncvarid, int nvdims, ncvar_t *ncvar)
   ncvar->hasFilter = cdf_get_var_filter(ncid, ncvarid, ncvar->filterSpec, CDI_MAX_NAME);
   // if (ncvar->hasFilter) printf("filterSpec: %s=%s\n", ncvar->name, ncvar->filterSpec);
 
-  size_t chunks[nvdims];
+  size_t chunks[MAX_DIMS_CDF];
   int storageIn;
   if (nc_inq_var_chunking(ncid, ncvarid, &storageIn, chunks) == NC_NOERR)
     {
@@ -1181,8 +1175,6 @@ set_vars_chunks(int ncid, int ncvarid, int nvdims, ncvar_t *ncvar)
               for (int i = 0; i < nvdims; ++i) fprintf(stderr, "%zu ", chunks[i]);
               fprintf(stderr, "\n");
             }
-
-          set_extra_attr(ncvar->extra, nvdims, chunks);
         }
     }
 
@@ -1245,10 +1237,9 @@ set_vars_timetype(int nvars, ncvar_t *ncvars, int timedimid)
             }
           else
             {
-              int nvdims = ncvar->ndims;
-              for (int vdimid = 1; vdimid < nvdims; vdimid++)
+              for (int i = 1, n = ncvar->ndims; i < n; i++)
                 {
-                  if (timedimid == ncvar->dimids[vdimid])
+                  if (timedimid == ncvar->dimids[i])
                     {
                       Warning("Time must be the first dimension! Unsupported array structure, skipped variable %s!", ncvar->name);
                       ncvar->varStatus = CoordVar;
@@ -1641,25 +1632,42 @@ scan_vars_attr(int nvars, ncvar_t *ncvars, int ndims, ncdim_t *ncdims, int model
 static void
 cdf_set_chunk_info(stream_t *streamptr, int nvars, ncvar_t *ncvars)
 {
+  int vlistID = streamptr->vlistID;
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
   for (int ncvarid = 0; ncvarid < nvars; ncvarid++)
     {
+      int chunkSizeDimT = 0;
+      int chunkSizeDimZ = 0;
+      int chunkSizeDimY = 0;
+      int chunkSizeDimX = 0;
       ncvar_t *ncvar = &ncvars[ncvarid];
-      if (ncvar->varStatus == DataVar)
+      int varID = ncvar->cdiVarID;
+      if (ncvar->varStatus == DataVar && ncvar->isChunked && varID != CDI_UNDEFID)
         {
-          int ndims = ncvar->ndims;
-          ncvar->isChunked = true;
-          for (int i = 0; i < ndims; ++i)
+          for (int i = 0, n = ncvar->ndims; i < n; ++i)
             {
               size_t chunkSize = ncvar->chunks[i];
               if (chunkSize > 1)
                 {
                   int dimType = ncvar->dimtypes[i];
                   // clang-format off
-                  if      (dimType == T_AXIS && chunkSize > streamptr->chunkSizeTdim) streamptr->chunkSizeTdim = chunkSize;
-                  else if (dimType == Z_AXIS && chunkSize > streamptr->chunkSizeZdim) streamptr->chunkSizeZdim = chunkSize;
-                  // clang-format on
+                  if      (dimType == T_AXIS && chunkSize > cdfInfo->chunkSizeDimT) cdfInfo->chunkSizeDimT = chunkSize;
+                  else if (dimType == Z_AXIS && chunkSize > cdfInfo->chunkSizeDimZ) cdfInfo->chunkSizeDimZ = chunkSize;
+
+                  if      (dimType == T_AXIS) chunkSizeDimT = chunkSize;
+                  else if (dimType == Z_AXIS) chunkSizeDimZ = chunkSize;
+                  else if (dimType == Y_AXIS) chunkSizeDimY = chunkSize;
+                  else if (dimType == X_AXIS) chunkSizeDimX = chunkSize;
+                  // clang-format onw
                 }
             }
+          if ((CDI_CopyChunkSpec || chunkSizeDimT == 0) && CDI_RemoveChunkSpec == false)
+            {
+              if (chunkSizeDimT > 0) cdiDefKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMT, chunkSizeDimT);
+              if (chunkSizeDimZ > 0) cdiDefKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMZ, chunkSizeDimZ);
+              if (chunkSizeDimY > 0) cdiDefKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMY, chunkSizeDimY);
+              if (chunkSizeDimX > 0) cdiDefKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE_DIMX, chunkSizeDimX);
+            }
         }
     }
 }
@@ -1688,9 +1696,9 @@ verify_vars_attr(int nvars, ncvar_t *ncvars, ncdim_t *ncdims)
           cdf_inq_atttype(ncid, ncvarid, attname, &atttype);
 
           size_t attstringsize = sizeof(attstring);
-          bool isText = xtypeIsText(atttype);
           // bool isNumber = (xtypeIsFloat(atttype) || xtypeIsInt(atttype));
 
+          bool isText = xtypeIsText(atttype);
           if (isText)
             {
               cdfGetAttText(ncid, ncvarid, attname, sizeof(attstring), attstring);
@@ -1717,8 +1725,7 @@ find_dimtypes(ncvar_t *ncvars, ncvar_t *ncvar, bool *plxdim, bool *plydim, bool
 {
   bool lxdim = false, lydim = false, lzdim = false /*, ltdim = false */;
   int lcdim = 0;
-  int ndims = ncvar->ndims;
-  for (int i = 0; i < ndims; i++)
+  for (int i = 0, n = ncvar->ndims; i < n; i++)
     {
       int dimtype = ncvar->dimtypes[i];
       lxdim |= (dimtype == X_AXIS);
@@ -1745,8 +1752,7 @@ cdf_set_dimtype(int nvars, ncvar_t *ncvars, ncdim_t *ncdims)
       ncvar_t *ncvar = &ncvars[varid];
       if (ncvar->varStatus == DataVar)
         {
-          int ndims = ncvar->ndims;
-          for (int i = 0; i < ndims; i++)
+          for (int i = 0, n = ncvar->ndims; i < n; i++)
             {
               int ncdimid = ncvar->dimids[i];
               int dimtype = ncdims[ncdimid].dimtype;
@@ -1756,7 +1762,7 @@ cdf_set_dimtype(int nvars, ncvar_t *ncvars, ncdim_t *ncdims)
           if (CDI_Debug)
             {
               Message("var %d %s", varid, ncvar->name);
-              for (int i = 0; i < ndims; i++) printf("  dim%d type=%d  ", i, ncvar->dimtypes[i]);
+              for (int i = 0, n = ncvar->ndims; i < n; i++) printf("  dim%d type=%d  ", i, ncvar->dimtypes[i]);
               printf("\n");
             }
         }
@@ -1797,8 +1803,7 @@ cdf_set_dimtype(int nvars, ncvar_t *ncvars, ncdim_t *ncdims)
   for (int varid = 0; varid < nvars; varid++)
     {
       ncvar_t *ncvar = &ncvars[varid];
-      int ndims = ncvar->ndims;
-      for (int i = 0; i < ndims; i++)
+      for (int i = 0, n = ncvar->ndims; i < n; i++)
         {
           if (ncvar->dimtypes[i] == CDI_UNDEFID)
             {
@@ -2115,7 +2120,7 @@ cdf_load_vals(size_t size, int ndims, int varid, ncvar_t *ncvar, double **gridva
 {
   if (CDI_Netcdf_Lazy_Grid_Load)
     {
-      *valsGet = (struct xyValGet){
+      *valsGet = (struct xyValGet) {
         .scalefactor = ncvar->scalefactor,
         .addoffset = ncvar->addoffset,
         .start = { start[0], start[1], start[2] },
@@ -2283,6 +2288,12 @@ cdf_get_xydimid(int ndims, int *dimids, int *dimtypes, int *xdimid, int *ydimid)
 static void
 cdf_check_gridtype(int *gridtype, bool isLon, bool isLat, size_t xsize, size_t ysize, grid_t *grid)
 {
+  if (grid->y.vals == NULL)
+    {
+      *gridtype = GRID_GENERIC;
+      return;
+    }
+
   if (isLat && (isLon || xsize == 0))
     {
       double yinc = 0.0;
@@ -2387,8 +2398,10 @@ cdf_read_xcoord(stream_t *streamptr, struct cdfLazyGrid *lazyGrid, ncdim_t *ncdi
       grid->x.clength = size / (*xsize);
 #endif
     }
-  else
-    cdf_load_vals(size, ndims, xvarid, axisvar, &grid->x.vals, &lazyGrid->xValsGet, hasTimeDim, readPart, start, count);
+  else if (CDI_Read_Cell_Center)
+    {
+      cdf_load_vals(size, ndims, xvarid, axisvar, &grid->x.vals, &lazyGrid->xValsGet, hasTimeDim, readPart, start, count);
+    }
 
   cdf_copy_grid_axis_attr(axisvar, &grid->x);
 
@@ -2469,8 +2482,10 @@ cdf_read_ycoord(stream_t *streamptr, struct cdfLazyGrid *lazyGrid, ncdim_t *ncdi
       grid->y.clength = size / (*ysize);
 #endif
     }
-  else
-    cdf_load_vals(size, ndims, yvarid, axisvar, &grid->y.vals, &lazyGrid->yValsGet, hasTimeDim, readPart, start, count);
+  else if (CDI_Read_Cell_Center)
+    {
+      cdf_load_vals(size, ndims, yvarid, axisvar, &grid->y.vals, &lazyGrid->yValsGet, hasTimeDim, readPart, start, count);
+    }
 
   cdf_copy_grid_axis_attr(axisvar, &grid->y);
 
@@ -2981,7 +2996,7 @@ destroy_grid(struct cdfLazyGrid *lazyGrid, grid_t *grid)
 }
 
 static int
-cdf_define_all_grids(stream_t *streamptr, ncgrid_t *ncgrid, int vlistID, ncdim_t *ncdims, int nvars, ncvar_t *ncvars,
+cdf_define_all_grids(stream_t *streamptr, CdfGrid *ncgrid, int vlistID, ncdim_t *ncdims, int nvars, ncvar_t *ncvars,
                      GridInfo *gridInfo)
 {
   for (int ncvarid = 0; ncvarid < nvars; ++ncvarid)
@@ -3155,14 +3170,14 @@ cdf_define_all_grids(stream_t *streamptr, ncgrid_t *ncgrid, int vlistID, ncdim_t
           ncgrid[gridindex].gridID = gridID;
           if (grid->type == GRID_TRAJECTORY)
             {
-              ncgrid[gridindex].ncIDs[CDF_VARID_X] = xvarid;
-              ncgrid[gridindex].ncIDs[CDF_VARID_Y] = yvarid;
+              ncgrid[gridindex].ncIdVec[CDF_VARID_X] = xvarid;
+              ncgrid[gridindex].ncIdVec[CDF_VARID_Y] = yvarid;
             }
           else
             {
-              if (xdimid != CDI_UNDEFID) ncgrid[gridindex].ncIDs[CDF_DIMID_X] = ncdims[xdimid].dimid;
-              if (ydimid != CDI_UNDEFID) ncgrid[gridindex].ncIDs[CDF_DIMID_Y] = ncdims[ydimid].dimid;
-              if (ncvar->isCubeSphere) ncgrid[gridindex].ncIDs[CDF_DIMID_E] = ncdims[ncvar->dimids[ndims - 3]].dimid;
+              if (xdimid != CDI_UNDEFID) ncgrid[gridindex].ncIdVec[CDF_DIMID_X] = ncdims[xdimid].dimid;
+              if (ydimid != CDI_UNDEFID) ncgrid[gridindex].ncIdVec[CDF_DIMID_Y] = ncdims[ydimid].dimid;
+              if (ncvar->isCubeSphere) ncgrid[gridindex].ncIdVec[CDF_DIMID_E] = ncdims[ncvar->dimids[ndims - 3]].dimid;
             }
 
           if (xdimid == CDI_UNDEFID && ydimid == CDI_UNDEFID && grid->size == 1) gridDefHasDims(gridID, CoordVar);
@@ -3261,12 +3276,12 @@ cdf_define_all_zaxes(stream_t *streamptr, int vlistID, ncdim_t *ncdims, int nvar
 	          if      (ncvars[zvarid].xtype == NC_FLOAT) zdatatype = CDI_DATATYPE_FLT32;
 	          else if (ncvars[zvarid].xtype == NC_INT)   zdatatype = CDI_DATATYPE_INT32;
 	          else if (ncvars[zvarid].xtype == NC_SHORT) zdatatype = CDI_DATATYPE_INT16;
-                // clang-format on
-                // don't change the name !!!
-                /*
-                  if ((len = strlen(pname)) > 2)
-                  if (pname[len-2] == '_' && isdigit((int) pname[len-1])) pname[len-2] = 0;
-                */
+              // clang-format on
+              // don't change the name !!!
+              /*
+                if ((len = strlen(pname)) > 2)
+                if (pname[len-2] == '_' && isdigit((int) pname[len-1])) pname[len-2] = 0;
+              */
 #ifndef USE_MPI
               if (zaxisType == ZAXIS_CHAR && ncvars[zvarid].ndims == 2)
                 {
@@ -3339,6 +3354,7 @@ cdf_define_all_zaxes(stream_t *streamptr, int vlistID, ncdim_t *ncdims, int nvar
 
           ncvar->zaxisID = varDefZaxis(vlistID, zaxisType, (int) zsize, zvar, (const char **) zcvals, zclength, lbounds, ubounds,
                                        (int) vctsize, vct, pname, plongname, punits, zdatatype, 1, 0, -1);
+          ncvar->zSize = zsize;
 
           int zaxisID = ncvar->zaxisID;
 
@@ -3386,7 +3402,7 @@ cdf_define_all_zaxes(stream_t *streamptr, int vlistID, ncdim_t *ncdims, int nvar
             }
 
           int zaxisindex = vlistZaxisIndex(vlistID, zaxisID);
-          streamptr->zaxisID[zaxisindex] = zdimid >= 0 ? ncdims[zdimid].dimid : zdimid;
+          streamptr->cdfInfo.zaxisIdVec[zaxisindex] = zdimid >= 0 ? ncdims[zdimid].dimid : zdimid;
 
           if (CDI_Debug) Message("zaxisID %d %d %s", zaxisID, ncvarid, ncvar->name);
 
@@ -3528,28 +3544,6 @@ cdf_define_institut_and_model_id(int vlistID, int varID)
   if (varTableID != CDI_UNDEFID) vlistDefVarTable(vlistID, varID, varTableID);
 }
 
-static size_t
-cdf_xtype_to_numbytes(int xtype)
-{
-  size_t numBytes = 8;
-
-  // clang-format off
-  if      (xtype == NC_BYTE  )  numBytes = 1;
-  else if (xtype == NC_CHAR  )  numBytes = 1;
-  else if (xtype == NC_SHORT )  numBytes = 2;
-  else if (xtype == NC_INT   )  numBytes = 4;
-  else if (xtype == NC_FLOAT )  numBytes = 4;
-#ifdef HAVE_NETCDF4
-  else if (xtype == NC_UBYTE )  numBytes = 1;
-  else if (xtype == NC_USHORT)  numBytes = 2;
-  else if (xtype == NC_LONG  )  numBytes = 4;
-  else if (xtype == NC_UINT  )  numBytes = 4;
-#endif
-  // clang-format on
-
-  return numBytes;
-}
-
 static inline size_t
 size_of_dim_chunks(size_t n, size_t c)
 {
@@ -3559,20 +3553,21 @@ size_of_dim_chunks(size_t n, size_t c)
 static size_t
 calc_chunk_cache_size(int timedimid, ncvar_t *ncvar)
 {
-  size_t nx = 0, ny = 0;
+  size_t nx = 0, ny = 0, nz = 0;
   size_t cx = 0, cy = 0, cz = 0;
   for (int i = 0; i < ncvar->ndims; i++)
     {
       int dimtype = ncvar->dimtypes[i];
       // clang-format off
-      if      (dimtype == Z_AXIS) { cz = ncvar->chunks[i]; }
+      if      (dimtype == Z_AXIS) { cz = ncvar->chunks[i]; nz = ncvar->zSize; }
       else if (dimtype == Y_AXIS) { cy = ncvar->chunks[i]; ny = ncvar->ySize; }
       else if (dimtype == X_AXIS) { cx = ncvar->chunks[i]; nx = ncvar->xSize; }
       // clang-format on
     }
 
-  size_t chunkCacheSize = (ncvar->dimids[0] == timedimid) ? ncvar->chunks[0] : 1;
-  if (cz > 0) chunkCacheSize *= cz;
+  size_t numSteps = (ncvar->dimids[0] == timedimid) ? ncvar->chunks[0] : 1;
+  size_t chunkCacheSize = numSteps;
+  if (nz > 0 && cz > 0) chunkCacheSize *= (numSteps == 1) ? cz : size_of_dim_chunks(nz, cz);
 
   if (chunkCacheSize == 1) return 0;  // no chunk cache needed because the full field is read
 
@@ -3617,6 +3612,7 @@ cdf_define_all_vars(stream_t *streamptr, int vlistID, int instID, int modelID, i
 
       stream_new_var(streamptr, gridID, zaxisID, CDI_UNDEFID);
       int varID = vlistDefVar(vlistID, gridID, zaxisID, ncvar->timetype);
+      ncvar->cdiVarID = varID;
 
 #ifdef HAVE_NETCDF4
       if (ncvar->hasFilter) cdiDefKeyString(vlistID, varID, CDI_KEY_FILTERSPEC_IN, ncvar->filterSpec);
@@ -3667,11 +3663,12 @@ cdf_define_all_vars(stream_t *streamptr, int vlistID, int instID, int modelID, i
         Message("varID = %d  gridID = %d  zaxisID = %d", varID, vlistInqVarGrid(vlistID, varID), vlistInqVarZaxis(vlistID, varID));
 
       int gridindex = vlistGridIndex(vlistID, gridID);
-      int xdimid = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_X];
-      int ydimid = streamptr->ncgrid[gridindex].ncIDs[CDF_DIMID_Y];
+      const CdfGrid *ncGrid = &(streamptr->cdfInfo.cdfGridVec[gridindex]);
+      int xdimid = ncGrid->ncIdVec[CDF_DIMID_X];
+      int ydimid = ncGrid->ncIdVec[CDF_DIMID_Y];
 
       int zaxisindex = vlistZaxisIndex(vlistID, zaxisID);
-      int zdimid = streamptr->zaxisID[zaxisindex];
+      int zdimid = streamptr->cdfInfo.zaxisIdVec[zaxisindex];
 
       int ndims = ncvar->ndims;
       static const int ipow10[4] = { 1, 10, 100, 1000 };
@@ -3714,8 +3711,6 @@ cdf_define_all_vars(stream_t *streamptr, int vlistID, int instID, int modelID, i
           if (ncvar->numberOfForecastsInEnsemble != -1)
             cdiDefKeyInt(vlistID, varID, CDI_KEY_TYPEOFENSEMBLEFORECAST, ncvar->typeOfEnsembleForecast);
         }
-
-      if (ncvar->extra[0] != 0) cdiDefKeyString(vlistID, varID, CDI_KEY_CHUNKS, ncvar->extra);
     }
 
   for (int varID = 0; varID < nvars; varID++)
@@ -4445,9 +4440,10 @@ static void
 stream_set_ncdims(stream_t *streamptr, int ndims, ncdim_t *ncdims)
 {
   int n = (ndims > MAX_DIMS_PS) ? MAX_DIMS_PS : ndims;
-  streamptr->ncNumDims = n;
-  for (int i = 0; i < n; i++) streamptr->ncDimID[i] = ncdims[i].dimid;
-  for (int i = 0; i < n; i++) streamptr->ncDimLen[i] = ncdims[i].len;
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  cdfInfo->ncNumDims = n;
+  for (int i = 0; i < n; i++) cdfInfo->ncDimIdVec[i] = ncdims[i].dimid;
+  for (int i = 0; i < n; i++) cdfInfo->ncDimLenVec[i] = ncdims[i].len;
 }
 
 static void
@@ -4722,7 +4718,7 @@ cdfInqContents(stream_t *streamptr)
 
   // define all grids
   gridInfo.timedimid = timedimid;
-  int status = cdf_define_all_grids(streamptr, streamptr->ncgrid, vlistID, ncdims, nvars, ncvars, &gridInfo);
+  int status = cdf_define_all_grids(streamptr, streamptr->cdfInfo.cdfGridVec, vlistID, ncdims, nvars, ncvars, &gridInfo);
   if (status < 0) return status;
 
   // define all zaxes
@@ -4731,8 +4727,7 @@ cdfInqContents(stream_t *streamptr)
   if (status < 0) return status;
 
   // verify vars
-  cdfVerifyVars(nvars, ncvars  //, ncdims
-  );
+  cdfVerifyVars(nvars, ncvars);
 
   // select vars
   int nvarsData = 0;
diff --git a/src/stream_cdf_o.c b/src/stream_cdf_o.c
index 987fb0514748e8247f2af414fa2003f1264e9217..22b50d769565f4d2ea0835c8de64e5b12c7400df 100644
--- a/src/stream_cdf_o.c
+++ b/src/stream_cdf_o.c
@@ -18,13 +18,13 @@
 static const char bndsName[] = "bnds";
 
 void
-cdfCopyRecord(stream_t *streamptr2, stream_t *streamptr1)
+cdfCopyField(stream_t *streamptr2, stream_t *streamptr1)
 {
   int vlistID1 = streamptr1->vlistID;
   int tsID = streamptr1->curTsID;
   int vrecID = streamptr1->tsteps[tsID].curRecID;
   int recID = streamptr1->tsteps[tsID].recIDs[vrecID];
-  int ivarID = streamptr1->tsteps[tsID].records[recID].varID;
+  int ivarID = streamptr1->tsteps[tsID].recinfo[recID].varID;
   int gridID = vlistInqVarGrid(vlistID1, ivarID);
   size_t datasize = (size_t) gridInqSize(gridID);
   int datatype = vlistInqVarDatatype(vlistID1, ivarID);
@@ -49,17 +49,17 @@ static void
 cdfDefComplex(stream_t *streamptr, int gridID, int gridIndex)
 {
   int dimID;
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
 
   for (int index = 0; index < gridIndex; ++index)
     {
-      if (ncgrid[index].ncIDs[CDF_DIMID_X] != CDI_UNDEFID)
+      if (cdfGridVec[index].ncIdVec[CDF_DIMID_X] != CDI_UNDEFID)
         {
-          int gridID0 = ncgrid[index].gridID;
+          int gridID0 = cdfGridVec[index].gridID;
           int gridtype0 = gridInqType(gridID0);
           if (gridtype0 == GRID_SPECTRAL || gridtype0 == GRID_FOURIER)
             {
-              dimID = ncgrid[index].ncIDs[CDF_DIMID_X];
+              dimID = cdfGridVec[index].ncIdVec[CDF_DIMID_X];
               goto dimIDEstablished;
             }
         }
@@ -87,8 +87,8 @@ cdfDefComplex(stream_t *streamptr, int gridID, int gridIndex)
   }
 
 dimIDEstablished:
-  ncgrid[gridIndex].gridID = gridID;
-  ncgrid[gridIndex].ncIDs[CDF_DIMID_X] = dimID;
+  cdfGridVec[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_X] = dimID;
 }
 
 struct idSearch
@@ -98,22 +98,22 @@ struct idSearch
 };
 
 static inline struct idSearch
-cdfSearchIDBySize(size_t startIdx, size_t numIDs, const ncgrid_t ncgrid[/*numIDs*/], int ncIDType, int searchType,
+cdfSearchIDBySize(size_t startIdx, size_t numIDs, const CdfGrid cdfGridVec[/*numIDs*/], int ncIDType, int searchType,
                   SizeType searchSize, int (*typeInq)(int id), SizeType (*sizeInq)(int id))
 {
   int numNonMatching = 0, foundID = CDI_UNDEFID;
   size_t foundIdx = SIZE_MAX;
   for (size_t index = startIdx; index < numIDs; index++)
     {
-      if (ncgrid[index].ncIDs[ncIDType] != CDI_UNDEFID)
+      if (cdfGridVec[index].ncIdVec[ncIDType] != CDI_UNDEFID)
         {
-          int id0 = ncgrid[index].gridID, id0Type = typeInq(id0);
+          int id0 = cdfGridVec[index].gridID, id0Type = typeInq(id0);
           if (id0Type == searchType)
             {
               SizeType size0 = sizeInq(id0);
               if (searchSize == size0)
                 {
-                  foundID = ncgrid[index].ncIDs[ncIDType];
+                  foundID = cdfGridVec[index].ncIdVec[ncIDType];
                   foundIdx = index;
                   break;
                 }
@@ -121,7 +121,7 @@ cdfSearchIDBySize(size_t startIdx, size_t numIDs, const ncgrid_t ncgrid[/*numIDs
             }
         }
     }
-  return (struct idSearch){ .numNonMatching = numNonMatching, .foundID = foundID, .foundIdx = foundIdx };
+  return (struct idSearch) { .numNonMatching = numNonMatching, .foundID = foundID, .foundIdx = foundIdx };
 }
 
 static SizeType
@@ -133,12 +133,12 @@ cdfGridInqHalfSize(int gridID)
 static void
 cdfDefSPorFC(stream_t *streamptr, int gridID, int gridIndex, char *axisname, size_t maxlen, int gridRefType)
 {
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
 
   SizeType dimlen = gridInqSize(gridID) / 2;
 
   struct idSearch search
-      = cdfSearchIDBySize(0, (size_t) gridIndex, ncgrid, CDF_DIMID_Y, gridRefType, dimlen, gridInqType, cdfGridInqHalfSize);
+      = cdfSearchIDBySize(0, (size_t) gridIndex, cdfGridVec, CDF_DIMID_Y, gridRefType, dimlen, gridInqType, cdfGridInqHalfSize);
   int dimID = search.foundID;
   int iz = search.numNonMatching;
 
@@ -164,8 +164,8 @@ cdfDefSPorFC(stream_t *streamptr, int gridID, int gridIndex, char *axisname, siz
         }
     }
 
-  ncgrid[gridIndex].gridID = gridID;
-  ncgrid[gridIndex].ncIDs[CDF_DIMID_Y] = dimID;
+  cdfGridVec[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_Y] = dimID;
 }
 
 static void
@@ -256,12 +256,12 @@ static void
 cdfDefTrajLatLon(stream_t *streamptr, int gridID, int gridIndex, const struct cdfDefGridAxisInqs *inqs)
 {
   nc_type xtype = grid_inq_xtype(gridID);
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *ncgrid = streamptr->cdfInfo.cdfGridVec;
 
   SizeType dimlen = inqs->axisSize(gridID);
   if (dimlen != 1) Error("%c size isn't 1 for %s grid!", inqs->axisSym, gridNamePtr(gridInqType(gridID)));
 
-  int ncvarid = ncgrid[gridIndex].ncIDs[inqs->dimIdx];
+  int ncvarid = ncgrid[gridIndex].ncIdVec[inqs->dimIdx];
   if (ncvarid == CDI_UNDEFID)
     {
       int dimNcID = streamptr->basetime.ncvarid;
@@ -290,7 +290,7 @@ cdfDefTrajLatLon(stream_t *streamptr, int gridID, int gridIndex, const struct cd
 
   ncgrid[gridIndex].gridID = gridID;
   // var ID for trajectory !!!
-  ncgrid[gridIndex].ncIDs[inqs->dimIdx] = ncvarid;
+  ncgrid[gridIndex].ncIdVec[inqs->dimIdx] = ncvarid;
 }
 
 static void
@@ -522,7 +522,7 @@ cdfPostDefActionAddPutVal(struct cdfPostDefActionList **list_, int fileID, int n
   delayedPutVals->fileID = fileID;
   delayedPutVals->ncvarid = ncvarid;
   *list_ = cdfPostDefActionAdd(
-      *list_, (struct cdfPostDefAction){ .data = (void *) delayedPutVals, .action = cdfDelayedPutVarDouble, .cleanup = cleanup });
+      *list_, (struct cdfPostDefAction) { .data = (void *) delayedPutVals, .action = cdfDelayedPutVarDouble, .cleanup = cleanup });
 }
 
 static inline void
@@ -533,7 +533,7 @@ cdfPostDefActionAddPut1Int(struct cdfPostDefActionList **list_, int fileID, int
   delayedPutVals->fileID = fileID;
   delayedPutVals->ncvarid = ncvarid;
   *list_ = cdfPostDefActionAdd(
-      *list_, (struct cdfPostDefAction){ .data = (void *) delayedPutVals, .action = cdfDelayedPutVarInt1, .cleanup = cleanup });
+      *list_, (struct cdfPostDefAction) { .data = (void *) delayedPutVals, .action = cdfDelayedPutVarInt1, .cleanup = cleanup });
 }
 
 static void
@@ -559,7 +559,7 @@ cdfDefAxisCommon(stream_t *streamptr, int gridID, int gridIndex, int ndims, bool
   SizeType dimlen = gridAxisInq->axisSize(gridID);
   nc_type xtype = grid_inq_xtype(gridID);
 
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
 
   bool hasVals = gridInqPropPresence(gridID, gridAxisInq->valsQueryKey);
   char dimname[CDI_MAX_NAME + 3];
@@ -569,7 +569,7 @@ cdfDefAxisCommon(stream_t *streamptr, int gridID, int gridIndex, int ndims, bool
 
   for (int index = 0; index < gridIndex; ++index)
     {
-      int gridID0 = ncgrid[index].gridID;
+      int gridID0 = cdfGridVec[index].gridID;
       assert(gridID0 != CDI_UNDEFID);
       int gridtype0 = gridInqType(gridID0);
       if (gridtype0 == GRID_GAUSSIAN || gridtype0 == GRID_LONLAT || gridtype0 == GRID_PROJECTION || gridtype0 == GRID_GENERIC)
@@ -586,7 +586,7 @@ cdfDefAxisCommon(stream_t *streamptr, int gridID, int gridIndex, int ndims, bool
               if (IS_EQUAL(inqVal(gridID0, 0), inqVal(gridID, 0))
                   && IS_EQUAL(inqVal(gridID0, dimlen - 1), inqVal(gridID, dimlen - 1)))
                 {
-                  dimID = ncgrid[index].ncIDs[(axisLetter == 'X') ? CDF_DIMID_X : CDF_DIMID_Y];
+                  dimID = cdfGridVec[index].ncIdVec[(axisLetter == 'X') ? CDF_DIMID_X : CDF_DIMID_Y];
                   break;
                 }
             }
@@ -691,11 +691,11 @@ cdfDefAxisCommon(stream_t *streamptr, int gridID, int gridIndex, int ndims, bool
           streamptr->ncmode = 2;
         }
 
-      if (ndims == 0 || addVarToGrid) ncgrid[gridIndex].ncIDs[(axisLetter == 'X') ? CDF_VARID_X : CDF_VARID_Y] = ncvarid;
+      if (ndims == 0 || addVarToGrid) cdfGridVec[gridIndex].ncIdVec[(axisLetter == 'X') ? CDF_VARID_X : CDF_VARID_Y] = ncvarid;
     }
 
-  ncgrid[gridIndex].gridID = gridID;
-  ncgrid[gridIndex].ncIDs[(axisLetter == 'X') ? CDF_DIMID_X : CDF_DIMID_Y] = dimID;
+  cdfGridVec[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].ncIdVec[(axisLetter == 'X') ? CDF_DIMID_X : CDF_DIMID_Y] = dimID;
 
   return delayed;
 }
@@ -930,7 +930,7 @@ cdfDefIrregularGridCommon(stream_t *streamptr, int gridID, size_t xsize, size_t
       streamptr->ncmode = 2;
     }
 
-  return (struct cdfDefIrregularGridCommonIDs){
+  return (struct cdfDefIrregularGridCommonIDs) {
     .xdimID = xdimID, .ydimID = ydimID, .ncxvarid = ncxvarid, .ncyvarid = ncyvarid, .ncavarid = ncavarid, .delayed = delayed
   };
 }
@@ -938,7 +938,7 @@ cdfDefIrregularGridCommon(stream_t *streamptr, int gridID, size_t xsize, size_t
 static struct cdfPostDefActionList *
 cdfDefCurvilinear(stream_t *streamptr, int gridID, int gridIndex)
 {
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
 
   SizeType dimlen = gridInqSize(gridID);
   SizeType xdimlen = gridInqXsize(gridID);
@@ -951,20 +951,20 @@ cdfDefCurvilinear(stream_t *streamptr, int gridID, int gridIndex)
   do
     {
       struct idSearch search
-          = cdfSearchIDBySize(ofs, (size_t) gridIndex, ncgrid, CDF_DIMID_X, GRID_CURVILINEAR, dimlen, gridInqType, gridInqSize);
+          = cdfSearchIDBySize(ofs, (size_t) gridIndex, cdfGridVec, CDF_DIMID_X, GRID_CURVILINEAR, dimlen, gridInqType, gridInqSize);
       size_t index = search.foundIdx;
       if (index != SIZE_MAX)
         {
-          int gridID0 = ncgrid[index].gridID;
+          int gridID0 = cdfGridVec[index].gridID;
           if (IS_EQUAL(gridInqXval(gridID0, 0), gridInqXval(gridID, 0))
               && IS_EQUAL(gridInqXval(gridID0, dimlen - 1), gridInqXval(gridID, dimlen - 1))
               && IS_EQUAL(gridInqYval(gridID0, 0), gridInqYval(gridID, 0))
               && IS_EQUAL(gridInqYval(gridID0, dimlen - 1), gridInqYval(gridID, dimlen - 1)))
             {
-              xdimID = ncgrid[index].ncIDs[CDF_DIMID_X];
-              ydimID = ncgrid[index].ncIDs[CDF_DIMID_Y];
-              ncxvarid = ncgrid[index].ncIDs[CDF_VARID_X];
-              ncyvarid = ncgrid[index].ncIDs[CDF_VARID_Y];
+              xdimID = cdfGridVec[index].ncIdVec[CDF_DIMID_X];
+              ydimID = cdfGridVec[index].ncIdVec[CDF_DIMID_Y];
+              ncxvarid = cdfGridVec[index].ncIdVec[CDF_VARID_X];
+              ncyvarid = cdfGridVec[index].ncIdVec[CDF_VARID_Y];
               break;
             }
           ofs = search.foundIdx;
@@ -987,19 +987,19 @@ cdfDefCurvilinear(stream_t *streamptr, int gridID, int gridIndex)
       delayed = createdIDs.delayed;
     }
 
-  ncgrid[gridIndex].gridID = gridID;
-  ncgrid[gridIndex].ncIDs[CDF_DIMID_X] = xdimID;
-  ncgrid[gridIndex].ncIDs[CDF_DIMID_Y] = ydimID;
-  ncgrid[gridIndex].ncIDs[CDF_VARID_X] = ncxvarid;
-  ncgrid[gridIndex].ncIDs[CDF_VARID_Y] = ncyvarid;
-  ncgrid[gridIndex].ncIDs[CDF_VARID_A] = ncavarid;
+  cdfGridVec[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_X] = xdimID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_Y] = ydimID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_VARID_X] = ncxvarid;
+  cdfGridVec[gridIndex].ncIdVec[CDF_VARID_Y] = ncyvarid;
+  cdfGridVec[gridIndex].ncIdVec[CDF_VARID_A] = ncavarid;
   return delayed;
 }
 
 static struct cdfPostDefActionList *
 cdfDefUnstructured(stream_t *streamptr, int gridID, int gridIndex)
 {
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
 
   SizeType dimlen = gridInqSize(gridID);
 
@@ -1009,21 +1009,21 @@ cdfDefUnstructured(stream_t *streamptr, int gridID, int gridIndex)
   size_t ofs = 0;
   do
     {
-      struct idSearch search
-          = cdfSearchIDBySize(ofs, (size_t) gridIndex, ncgrid, CDF_DIMID_X, GRID_UNSTRUCTURED, dimlen, gridInqType, gridInqSize);
+      struct idSearch search = cdfSearchIDBySize(ofs, (size_t) gridIndex, cdfGridVec, CDF_DIMID_X, GRID_UNSTRUCTURED, dimlen,
+                                                 gridInqType, gridInqSize);
       size_t index = search.foundIdx;
       if (index != SIZE_MAX)
         {
-          int gridID0 = ncgrid[index].gridID;
+          int gridID0 = cdfGridVec[index].gridID;
           if (gridInqNvertex(gridID0) == gridInqNvertex(gridID) && IS_EQUAL(gridInqXval(gridID0, 0), gridInqXval(gridID, 0))
               && IS_EQUAL(gridInqXval(gridID0, dimlen - 1), gridInqXval(gridID, dimlen - 1))
               && IS_EQUAL(gridInqYval(gridID0, 0), gridInqYval(gridID, 0))
               && IS_EQUAL(gridInqYval(gridID0, dimlen - 1), gridInqYval(gridID, dimlen - 1)))
             {
-              dimID = ncgrid[index].ncIDs[CDF_DIMID_X];
-              ncxvarid = ncgrid[index].ncIDs[CDF_VARID_X];
-              ncyvarid = ncgrid[index].ncIDs[CDF_VARID_Y];
-              ncavarid = ncgrid[index].ncIDs[CDF_VARID_A];
+              dimID = cdfGridVec[index].ncIdVec[CDF_DIMID_X];
+              ncxvarid = cdfGridVec[index].ncIdVec[CDF_VARID_X];
+              ncyvarid = cdfGridVec[index].ncIdVec[CDF_VARID_Y];
+              ncavarid = cdfGridVec[index].ncIdVec[CDF_VARID_A];
               break;
             }
           ofs = search.foundIdx;
@@ -1045,11 +1045,11 @@ cdfDefUnstructured(stream_t *streamptr, int gridID, int gridIndex)
       delayed = createdIDs.delayed;
     }
 
-  ncgrid[gridIndex].gridID = gridID;
-  ncgrid[gridIndex].ncIDs[CDF_DIMID_X] = dimID;
-  ncgrid[gridIndex].ncIDs[CDF_VARID_X] = ncxvarid;
-  ncgrid[gridIndex].ncIDs[CDF_VARID_Y] = ncyvarid;
-  ncgrid[gridIndex].ncIDs[CDF_VARID_A] = ncavarid;
+  cdfGridVec[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_X] = dimID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_VARID_X] = ncxvarid;
+  cdfGridVec[gridIndex].ncIdVec[CDF_VARID_Y] = ncyvarid;
+  cdfGridVec[gridIndex].ncIdVec[CDF_VARID_A] = ncavarid;
   return delayed;
 }
 
@@ -1075,10 +1075,10 @@ cdf_def_vct_echam(stream_t *streamptr, int zaxisID)
   if ((type == ZAXIS_HYBRID || type == ZAXIS_HYBRID_HALF) && (ilev = zaxisInqVctSize(zaxisID) / 2) != 0)
     {
       int mlev = ilev - 1;
-
-      if (streamptr->vct.ilev > 0)
+      CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+      if (cdfInfo->vct.ilev > 0)
         {
-          if (streamptr->vct.ilev != ilev) Error("More than one VCT for each file unsupported!");
+          if (cdfInfo->vct.ilev != ilev) Error("More than one VCT for each file unsupported!");
           return delayed;
         }
 
@@ -1104,10 +1104,10 @@ cdf_def_vct_echam(stream_t *streamptr, int zaxisID)
           cdf_def_var(fileID, "hybm", NC_DOUBLE, 1, &ncdimid, &hybmid);
         }
 
-      streamptr->vct.ilev = ilev;
-      streamptr->vct.mlev = mlev;
-      streamptr->vct.mlevID = ncdimid;
-      streamptr->vct.ilevID = ncdimid2;
+      cdfInfo->vct.ilev = ilev;
+      cdfInfo->vct.mlev = mlev;
+      cdfInfo->vct.mlevID = ncdimid;
+      cdfInfo->vct.ilevID = ncdimid2;
 
       {
         static const char lname_n[] = "long_name", units_n[] = "units", lname_v_ai[] = "hybrid A coefficient at layer interfaces",
@@ -1177,10 +1177,10 @@ cdf_def_vct_cf(stream_t *streamptr, int zaxisID, int nclevID, int ncbndsID, int
   if ((type == ZAXIS_HYBRID || type == ZAXIS_HYBRID_HALF) && (ilev = zaxisInqVctSize(zaxisID) / 2) != 0)
     {
       int mlev = ilev - 1;
-
-      if (streamptr->vct.ilev > 0)
+      CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+      if (cdfInfo->vct.ilev > 0)
         {
-          if (streamptr->vct.ilev != ilev) Error("more than one VCT for each file unsupported!");
+          if (cdfInfo->vct.ilev != ilev) Error("more than one VCT for each file unsupported!");
           return delayed;
         }
 
@@ -1195,10 +1195,10 @@ cdf_def_vct_cf(stream_t *streamptr, int zaxisID, int nclevID, int ncbndsID, int
 
       int dimIDs[2] = { nclevID, ncbndsID };
 
-      streamptr->vct.mlev = mlev;
-      streamptr->vct.ilev = ilev;
-      streamptr->vct.mlevID = nclevID;
-      streamptr->vct.ilevID = nclevID;
+      cdfInfo->vct.mlev = mlev;
+      cdfInfo->vct.ilev = ilev;
+      cdfInfo->vct.mlevID = nclevID;
+      cdfInfo->vct.ilevID = nclevID;
 
       int hyamid, hybmid;
       cdf_def_var(fileID, (p0status == 0) ? "a" : "ap", NC_DOUBLE, 1, dimIDs, &hyamid);
@@ -1350,7 +1350,8 @@ cdf_def_zaxis_hybrid_echam(stream_t *streamptr, int type, int *ncvaridp, int zax
     Free(delayedVct);
   }
 
-  if (*dimID == CDI_UNDEFID) streamptr->zaxisID[zaxisindex] = type == ZAXIS_HYBRID ? streamptr->vct.mlevID : streamptr->vct.ilevID;
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  if (*dimID == CDI_UNDEFID) cdfInfo->zaxisIdVec[zaxisindex] = (type == ZAXIS_HYBRID) ? cdfInfo->vct.mlevID : cdfInfo->vct.ilevID;
 
   if (switchNCMode)
     {
@@ -1550,7 +1551,8 @@ cdf_def_zaxis_hybrid_cf(stream_t *streamptr, int type, int *ncvaridp, int zaxisI
     Free(delayedVct);
   }
 
-  if (*dimID == CDI_UNDEFID) streamptr->zaxisID[zaxisindex] = type == ZAXIS_HYBRID ? streamptr->vct.mlevID : streamptr->vct.ilevID;
+  CdfInfo *cdfInfo = &(streamptr->cdfInfo);
+  if (*dimID == CDI_UNDEFID) cdfInfo->zaxisIdVec[zaxisindex] = (type == ZAXIS_HYBRID) ? cdfInfo->vct.mlevID : cdfInfo->vct.ilevID;
 
   free(buffer);
   return delayed;
@@ -1560,7 +1562,7 @@ static struct cdfPostDefActionList *
 cdf_def_zaxis_hybrid(stream_t *streamptr, int type, int *ncvarid, int zaxisID, int zaxisindex, int xtype, size_t dimlen, int *dimID,
                      char *axisname)
 {
-  struct cdfPostDefActionList *(*def_zaxis_hybrid_delegate)(stream_t * streamptr, int type, int *ncvarid, int zaxisID,
+  struct cdfPostDefActionList *(*def_zaxis_hybrid_delegate)(stream_t *streamptr, int type, int *ncvarid, int zaxisID,
                                                             int zaxisindex, int xtype, size_t dimlen, int *dimID, char *axisname)
       = ((!CDI_CMOR_Mode && CDI_Convention == CDI_CONVENTION_ECHAM) || type == ZAXIS_HYBRID_HALF) ? cdf_def_zaxis_hybrid_echam
                                                                                                   : cdf_def_zaxis_hybrid_cf;
@@ -1646,7 +1648,7 @@ cdfDefZaxisChar(stream_t *streamptr, int zaxisID, char *axisname, int *dimID, si
       cdf_put_att_text(fileID, ncvarID, "axis", 1, "Z");
       cdfDefineAttributes(streamptr->filetype, zaxisID, CDI_GLOBAL, fileID, ncvarID);
 
-      streamptr->nczvarID[zaxisindex] = ncvarID;
+      streamptr->cdfInfo.ncZvarIdVec[zaxisindex] = ncvarID;
       cdf_enddef(fileID, streamptr->self);
 
       // Write Stringvalues
@@ -1827,10 +1829,10 @@ cdfDefZaxis(stream_t *streamptr, int zaxisID)
           streamptr->ncmode = 2;
         }
 
-      if (zaxisInqLevels(zaxisID, NULL) && ndims == 0) streamptr->nczvarID[zaxisindex] = ncvarid;
+      if (zaxisInqLevels(zaxisID, NULL) && ndims == 0) streamptr->cdfInfo.ncZvarIdVec[zaxisindex] = ncvarid;
     }
 
-  if (dimID != CDI_UNDEFID) streamptr->zaxisID[zaxisindex] = dimID;
+  if (dimID != CDI_UNDEFID) streamptr->cdfInfo.zaxisIdVec[zaxisindex] = dimID;
   return delayed;
 }
 
@@ -1881,18 +1883,18 @@ cdf_def_mapping(stream_t *streamptr, int gridID)
 static void
 cdfDefCharacter(stream_t *streamptr, int gridID, int gridIndex, int cdiAxisID, int strlen)
 {
-  if (streamptr->ncgrid[gridIndex].ncIDs[CDF_DIMID_X] != CDI_UNDEFID) return;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
+  if (cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_X] != CDI_UNDEFID) return;
 
   bool isXaxis = (cdiAxisID == CDI_XAXIS);
 
   SizeType dimlen = isXaxis ? gridInqXsize(gridID) : gridInqYsize(gridID);
-  ncgrid_t *ncgrid = streamptr->ncgrid;
 
   // Check for all grids up to gridIndex whether it already is defined
 
   for (int index = 0; index < gridIndex; index++)
     {
-      int gridID0 = ncgrid[index].gridID;
+      int gridID0 = cdfGridVec[index].gridID;
       int gridtype0 = gridInqType(gridID0);
       if (gridtype0 == GRID_CHARXY)
         {
@@ -1957,9 +1959,9 @@ cdfDefCharacter(stream_t *streamptr, int gridID, int gridIndex, int cdiAxisID, i
       (void) nc_put_vara_text(fileID, ncaxisid, start, count, cvals[i]);
     }
 
-  ncgrid[gridIndex].gridID = gridID;
-  ncgrid[gridIndex].ncIDs[isXaxis ? CDF_DIMID_X : CDF_DIMID_Y] = dimID;
-  ncgrid[gridIndex].ncIDs[isXaxis ? CDF_VARID_X : CDF_VARID_Y] = ncaxisid;
+  cdfGridVec[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].ncIdVec[isXaxis ? CDF_DIMID_X : CDF_DIMID_Y] = dimID;
+  cdfGridVec[gridIndex].ncIdVec[isXaxis ? CDF_VARID_X : CDF_VARID_Y] = ncaxisid;
 
   streamptr->ncmode = 2;
 }
@@ -1968,15 +1970,15 @@ cdfDefCharacter(stream_t *streamptr, int gridID, int gridIndex, int cdiAxisID, i
 static void
 cdfDefReducedGrid(stream_t *streamptr, int gridID, int gridIndex)
 {
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
 
-  ncgrid[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].gridID = gridID;
 
   {
     SizeType dimlen = gridInqSize(gridID);
 
-    struct idSearch search
-        = cdfSearchIDBySize(0, (size_t) gridIndex, ncgrid, CDF_DIMID_X, GRID_GAUSSIAN_REDUCED, dimlen, gridInqType, gridInqSize);
+    struct idSearch search = cdfSearchIDBySize(0, (size_t) gridIndex, cdfGridVec, CDF_DIMID_X, GRID_GAUSSIAN_REDUCED, dimlen,
+                                               gridInqType, gridInqSize);
     int iz = search.numNonMatching;
     int dimID = search.foundID;
 
@@ -2004,14 +2006,14 @@ cdfDefReducedGrid(stream_t *streamptr, int gridID, int gridIndex)
           }
       }
 
-    ncgrid[gridIndex].ncIDs[CDF_DIMID_X] = dimID;
+    cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_X] = dimID;
   }
 
   {
     SizeType dimlen = gridInqYsize(gridID);
 
-    struct idSearch search
-        = cdfSearchIDBySize(0, (size_t) gridIndex, ncgrid, CDF_DIMID_RP, GRID_GAUSSIAN_REDUCED, dimlen, gridInqType, gridInqSize);
+    struct idSearch search = cdfSearchIDBySize(0, (size_t) gridIndex, cdfGridVec, CDF_DIMID_RP, GRID_GAUSSIAN_REDUCED, dimlen,
+                                               gridInqType, gridInqSize);
     int iz = search.numNonMatching;
     int dimID = search.foundID;
 
@@ -2038,17 +2040,17 @@ cdfDefReducedGrid(stream_t *streamptr, int gridID, int gridIndex)
         cdf_put_var_int(fileID, ncvarid, reducedPoints);
         Free(reducedPoints);
 
-        ncgrid[gridIndex].ncIDs[CDF_VARID_RP] = ncvarid;
+        cdfGridVec[gridIndex].ncIdVec[CDF_VARID_RP] = ncvarid;
       }
 
-    ncgrid[gridIndex].ncIDs[CDF_DIMID_RP] = dimID;
+    cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_RP] = dimID;
   }
 }
 
 static void
 cdf_define_generic_dim(stream_t *streamptr, int gridID, int gridIndex)
 {
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *cdfGridVec = streamptr->cdfInfo.cdfGridVec;
   int dimID = CDI_UNDEFID;
 
   SizeType dimlen = gridInqSize(gridID);
@@ -2056,14 +2058,14 @@ cdf_define_generic_dim(stream_t *streamptr, int gridID, int gridIndex)
   if (gridInqYsize(gridID) == 0)
     {
       struct idSearch search
-          = cdfSearchIDBySize(0, (size_t) gridIndex, ncgrid, CDF_DIMID_X, GRID_GENERIC, dimlen, gridInqType, gridInqSize);
+          = cdfSearchIDBySize(0, (size_t) gridIndex, cdfGridVec, CDF_DIMID_X, GRID_GENERIC, dimlen, gridInqType, gridInqSize);
       dimID = search.foundID;
     }
 
   if (gridInqXsize(gridID) == 0)
     {
       struct idSearch search
-          = cdfSearchIDBySize(0, (size_t) gridIndex, ncgrid, CDF_DIMID_Y, GRID_GENERIC, dimlen, gridInqType, gridInqSize);
+          = cdfSearchIDBySize(0, (size_t) gridIndex, cdfGridVec, CDF_DIMID_Y, GRID_GENERIC, dimlen, gridInqType, gridInqSize);
       dimID = search.foundID;
     }
 
@@ -2096,8 +2098,8 @@ cdf_define_generic_dim(stream_t *streamptr, int gridID, int gridIndex)
         }
     }
 
-  ncgrid[gridIndex].gridID = gridID;
-  ncgrid[gridIndex].ncIDs[CDF_DIMID_X] = dimID;
+  cdfGridVec[gridIndex].gridID = gridID;
+  cdfGridVec[gridIndex].ncIdVec[CDF_DIMID_X] = dimID;
 }
 
 static struct cdfPostDefActionList *
@@ -2105,7 +2107,8 @@ cdf_define_grid(stream_t *streamptr, int gridID, int gridIndex)
 {
   struct cdfPostDefActionList *delayed = NULL;
 
-  if (streamptr->ncgrid[gridIndex].ncIDs[CDF_DIMID_X] != CDI_UNDEFID) return delayed;
+  CdfGrid *ncgrid = &(streamptr->cdfInfo.cdfGridVec[gridIndex]);
+  if (ncgrid->ncIdVec[CDF_DIMID_X] != CDI_UNDEFID) return delayed;
 
   int gridtype = gridInqType(gridID);
   SizeType size = gridInqSize(gridID);
@@ -2114,7 +2117,7 @@ cdf_define_grid(stream_t *streamptr, int gridID, int gridIndex)
 
   if (CDI_Reduce_Dim && size == 1)  // no grid information
     {
-      streamptr->ncgrid[gridIndex].gridID = gridID;
+      ncgrid->gridID = gridID;
       return delayed;
     }
 
@@ -2151,7 +2154,7 @@ cdf_define_grid(stream_t *streamptr, int gridID, int gridIndex)
       if (size == 1 && xsize == 0 && ysize == 0)
         {
           // no grid information
-          streamptr->ncgrid[gridIndex].gridID = gridID;
+          ncgrid->gridID = gridID;
         }
       else
         {
@@ -2248,11 +2251,11 @@ cdfDefCoordinateVars(stream_t *streamptr)
 
   struct cdfPostDefActionList *delayed = NULL;
 
-  ncgrid_t *ncgrid = streamptr->ncgrid;
+  CdfGrid *ncgrid = streamptr->cdfInfo.cdfGridVec;
   for (int index = 0; index < 2 * ngrids; ++index)
     {
       ncgrid[index].gridID = CDI_UNDEFID;
-      for (size_t i = 0; i < CDF_SIZE_ncIDs; ++i) ncgrid[index].ncIDs[i] = CDI_UNDEFID;
+      for (size_t i = 0; i < CDF_SIZE_NCID; ++i) ncgrid[index].ncIdVec[i] = CDI_UNDEFID;
     }
 
   for (int index = 0; index < ngrids; ++index)
@@ -2281,7 +2284,7 @@ cdfDefCoordinateVars(stream_t *streamptr)
   for (int index = 0; index < nzaxis; ++index)
     {
       int zaxisID = vlistZaxis(vlistID, index);
-      if (streamptr->zaxisID[index] == CDI_UNDEFID)
+      if (streamptr->cdfInfo.zaxisIdVec[index] == CDI_UNDEFID)
         {
           struct cdfPostDefActionList *zaxisdelayed = cdfDefZaxis(streamptr, zaxisID);
           delayed = cdfPostDefActionConcat(delayed, zaxisdelayed);
diff --git a/src/stream_cdf_time.c b/src/stream_cdf_time.c
index dc594c44d4d14dd8080fc484636b7cdff7f8ecfb..1a9adac28daffac1c1e89c7987129fbaec9fc31b 100644
--- a/src/stream_cdf_time.c
+++ b/src/stream_cdf_time.c
@@ -10,10 +10,8 @@
 
 #include "cdi.h"
 #include "cdi_int.h"
-#include "dmemory.h"
 #include "stream_cdf.h"
 #include "cdf_int.h"
-#include "vlist.h"
 
 static int
 cdfDefTimeBounds(int fileID, int nctimevarid, int nctimedimid, const char *taxis_name, taxis_t *taxis)
@@ -153,7 +151,7 @@ cdfDefTime(stream_t *streamptr)
   if (streamptr->ncmode == 0) streamptr->ncmode = 1;
   if (streamptr->ncmode == 2) cdf_redef(fileID);
 
-  taxis_t *taxis = taxisPtr(vlistInqTaxis(streamptr->vlistID));
+  taxis_t *taxis = taxis_to_pointer(vlistInqTaxis(streamptr->vlistID));
 
   const char *taxisName = (taxis->name && taxis->name[0]) ? taxis->name : defaultTimeAxisName;
 
@@ -170,7 +168,8 @@ cdfDefTime(stream_t *streamptr)
   cdf_def_dim(fileID, taxisName, timeDimLen, &timeDimId);
   streamptr->basetime.ncdimid = timeDimId;
 
-  int datatype = taxis->datatype;
+  int datatype = CDI_UNDEFID;
+  cdiInqKeyInt(taxis->self, CDI_GLOBAL, CDI_KEY_DATATYPE, &datatype);
   nc_type xtype = (datatype == CDI_DATATYPE_INT32) ? NC_INT : ((datatype == CDI_DATATYPE_FLT32) ? NC_FLOAT : NC_DOUBLE);
 
   int timeVarId;
@@ -245,7 +244,7 @@ cdfDefTimestep(stream_t *streamptr, int tsID, size_t valCount)
   int time_varid = streamptr->basetime.ncvarid;
   if (time_varid != CDI_UNDEFID && tsID == 0)
     {
-      taxis_t *taxis = taxisPtr(vlistInqTaxis(streamptr->vlistID));
+      taxis_t *taxis = taxis_to_pointer(vlistInqTaxis(streamptr->vlistID));
       int fileID = streamptr->fileID;
       const char *unitstr = cdfGetTimeUnits(taxis);
       size_t len = strlen(unitstr);
diff --git a/src/stream_cgribex.c b/src/stream_cgribex.c
index a2c73b7f0c104ce5fc2657e94d659287bcd380b2..94d9e134db3968894ab9b1fcdf23c45f6861e5fd 100644
--- a/src/stream_cgribex.c
+++ b/src/stream_cgribex.c
@@ -14,7 +14,7 @@
 #include "stream_grb.h"
 #include "stream_cgribex.h"
 
-#ifdef HAVE_LIBCGRIBEX
+#if defined(HAVE_LIBCGRIBEX) && defined(HAVE_LIBGRIB)
 
 #include "cgribex.h"
 
@@ -32,12 +32,12 @@ typedef struct
 
 typedef struct
 {
+  size_t gridsize;
   int param;
   int level1;
   int level2;
-  int ltype;
-  int tsteptype;
-  size_t gridsize;
+  short ltype;
+  short tsteptype;
 } compvar_t;
 
 typedef struct
@@ -548,7 +548,8 @@ cgribexAddRecord(stream_t *streamptr, cgribexrec_t *cgribexp, int param, size_t
   int vlistID = streamptr->vlistID;
   int tsID = streamptr->curTsID;
   int recID = recordNewEntry(streamptr, tsID);
-  record_t *record = &streamptr->tsteps[tsID].records[recID];
+  recinfo_t *recinfo = &(streamptr->tsteps[tsID].recinfo[recID]);
+  record_t *record = &(streamptr->tsteps[tsID].records[recID]);
 
   int tsteptype = cgribexGetTsteptype(ISEC1_TimeRange);
 
@@ -562,7 +563,7 @@ cgribexAddRecord(stream_t *streamptr, cgribexrec_t *cgribexp, int param, size_t
   record->param = param;
   record->ilevel = level1;
   record->ilevel2 = level2;
-  record->ltype = leveltype;
+  record->ltype = (short) leveltype;
   record->tsteptype = (short) tsteptype;
   record->gridsize = cgribexGetGridsize(cgribexp->sec4);
 
@@ -602,8 +603,8 @@ cgribexAddRecord(stream_t *streamptr, cgribexrec_t *cgribexp, int param, size_t
   varAddRecord(recID, param, gridID, zaxistype, lbounds, level1, level2, 0, 0, datatype, &varID, &levelID, tsteptype, leveltype, -1,
                NULL, NULL, NULL, NULL);
 
-  record->varID = (short) varID;
-  record->levelID = levelID;
+  recinfo->varID = (short) varID;
+  recinfo->levelID = levelID;
 
   varDefCompType(varID, comptype);
 
@@ -717,8 +718,8 @@ cgribexVarSet(int param, int level1, int level2, int leveltype, int trange, size
   compVar.param = param;
   compVar.level1 = level1;
   compVar.level2 = level2;
-  compVar.ltype = leveltype;
-  compVar.tsteptype = tsteptype;
+  compVar.ltype = (short) leveltype;
+  compVar.tsteptype = (short) tsteptype;
   compVar.gridsize = gridsize;
 
   return compVar;
@@ -958,7 +959,8 @@ cgribexScanTimestep2(stream_t *streamptr)
 
   fileSetPos(fileID, streamptr->tsteps[tsID].position, SEEK_SET);
 
-  cdi_create_records(streamptr, tsID);
+  cdi_create_records(streamptr, tsID, true);
+  recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
   record_t *records = streamptr->tsteps[tsID].records;
 
   int nrecords = streamScanInitRecords2(streamptr);
@@ -1044,19 +1046,19 @@ cgribexScanTimestep2(stream_t *streamptr)
 
       if (CDI_Inventory_Mode == 1)
         {
-          if (records[recID].used)
+          if (recinfo[recID].used)
             {
               break;
             }
           else
             {
-              records[recID].used = true;
+              recinfo[recID].used = true;
               streamptr->tsteps[tsID].recIDs[rindex] = recID;
             }
         }
       else
         {
-          if (records[recID].used)
+          if (recinfo[recID].used)
             {
               if (cdiDateTime_isNE(vDateTime, vDateTime0)) break;
 
@@ -1065,7 +1067,7 @@ cgribexScanTimestep2(stream_t *streamptr)
             }
           else
             {
-              records[recID].used = true;
+              recinfo[recID].used = true;
               streamptr->tsteps[tsID].recIDs[rindex] = recID;
             }
         }
@@ -1084,7 +1086,7 @@ cgribexScanTimestep2(stream_t *streamptr)
       records[recID].position = recpos;
       records[recID].size = recsize;
 
-      int varID = records[recID].varID;
+      int varID = recinfo[recID].varID;
       int gridID = vlistInqVarGrid(vlistID, varID);
       if (gridInqSize(gridID) == 1 && gridInqType(gridID) == GRID_LONLAT)
         {
@@ -1102,10 +1104,10 @@ cgribexScanTimestep2(stream_t *streamptr)
   int nrecs = 0;
   for (recID = 0; recID < nrecords; recID++)
     {
-      if (records[recID].used)
+      if (recinfo[recID].used)
         nrecs++;
       else
-        vlistDefVarTimetype(vlistID, records[recID].varID, TIME_CONSTANT);
+        vlistDefVarTimetype(vlistID, recinfo[recID].varID, TIME_CONSTANT);
     }
   streamptr->tsteps[tsID].nrecs = nrecs;
 
@@ -1143,7 +1145,8 @@ cgribexScanTimestep(stream_t *streamptr)
       void *gribbuffer = streamptr->record->buffer;
       size_t buffersize = streamptr->record->buffersize;
 
-      cdi_create_records(streamptr, tsID);
+      cdi_create_records(streamptr, tsID, true);
+      recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
       record_t *records = streamptr->tsteps[tsID].records;
 
       nrecs = streamScanInitRecords(streamptr, tsID);
@@ -1243,12 +1246,12 @@ cgribexScanTimestep(stream_t *streamptr)
 
           if (CDI_Inventory_Mode == 1)
             {
-              records[recID].used = true;
+              recinfo[recID].used = true;
               streamptr->tsteps[tsID].recIDs[rindex] = recID;
             }
           else
             {
-              if (records[recID].used)
+              if (recinfo[recID].used)
                 {
                   char paramstr_[32];
                   cdiParamToString(param, paramstr_, sizeof(paramstr_));
@@ -1262,7 +1265,7 @@ cgribexScanTimestep(stream_t *streamptr)
                 }
               else
                 {
-                  records[recID].used = true;
+                  recinfo[recID].used = true;
                   streamptr->tsteps[tsID].recIDs[rindex] = recID;
                 }
             }
@@ -1287,7 +1290,7 @@ cgribexScanTimestep(stream_t *streamptr)
       for (vrecID = 0; vrecID < nrecs; vrecID++)
         {
           recID = streamptr->tsteps[tsID].recIDs[vrecID];
-          if (!records[recID].used) break;
+          if (!recinfo[recID].used) break;
         }
 
       if (vrecID < nrecs)
diff --git a/src/stream_ext.c b/src/stream_ext.c
index 616ef92dc08043afedb8b2f06dc9aec377039ef0..a5cd4f5aeda29c2a26d3ef0d709ffc5cfc9f92d1 100644
--- a/src/stream_ext.c
+++ b/src/stream_ext.c
@@ -47,7 +47,7 @@ ext_read_recordSP(stream_t *streamptr, float *data, size_t *numMissVals)
 
   int vrecID = streamptr->tsteps[tsID].curRecID;
   int recID = streamptr->tsteps[tsID].recIDs[vrecID];
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
+  int varID = streamptr->tsteps[tsID].recinfo[recID].varID;
   off_t recpos = streamptr->tsteps[tsID].records[recID].position;
 
   fileSetPos(fileID, recpos, SEEK_SET);
@@ -79,7 +79,7 @@ ext_read_recordDP(stream_t *streamptr, double *data, size_t *numMissVals)
 
   int vrecID = streamptr->tsteps[tsID].curRecID;
   int recID = streamptr->tsteps[tsID].recIDs[vrecID];
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
+  int varID = streamptr->tsteps[tsID].recinfo[recID].varID;
   off_t recpos = streamptr->tsteps[tsID].records[recID].position;
 
   fileSetPos(fileID, recpos, SEEK_SET);
@@ -169,7 +169,8 @@ extAddRecord(stream_t *streamptr, int param, int level, size_t xysize, size_t re
   int vlistID = streamptr->vlistID;
   int tsID = streamptr->curTsID;
   int recID = recordNewEntry(streamptr, tsID);
-  record_t *record = &streamptr->tsteps[tsID].records[recID];
+  recinfo_t *recinfo = &(streamptr->tsteps[tsID].recinfo[recID]);
+  record_t *record = &(streamptr->tsteps[tsID].records[recID]);
 
   record->size = recsize;
   record->position = position;
@@ -196,8 +197,8 @@ extAddRecord(stream_t *streamptr, int param, int level, size_t xysize, size_t re
   varAddRecord(recID, param, gridID, leveltype, 0, level, 0, 0, 0, datatype, &varID, &levelID, TSTEP_INSTANT, 0, -1, NULL, NULL,
                NULL, NULL);
 
-  record->varID = (short) varID;
-  record->levelID = levelID;
+  recinfo->varID = (short) varID;
+  recinfo->levelID = levelID;
 
   streamptr->tsteps[tsID].nallrecs++;
   streamptr->nrecs++;
@@ -304,7 +305,8 @@ extScanTimestep2(stream_t *streamptr)
 
   fileSetPos(fileID, streamptr->tsteps[tsID].position, SEEK_SET);
 
-  cdi_create_records(streamptr, tsID);
+  cdi_create_records(streamptr, tsID, true);
+  recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
   record_t *records = streamptr->tsteps[tsID].records;
 
   int nrecords = streamScanInitRecords2(streamptr);
@@ -340,13 +342,13 @@ extScanTimestep2(stream_t *streamptr)
         {
           if (param == records[recID].param && rlevel == records[recID].ilevel)
             {
-              if (records[recID].used)
+              if (recinfo[recID].used)
                 {
                   nextstep = true;
                 }
               else
                 {
-                  records[recID].used = true;
+                  recinfo[recID].used = true;
                   streamptr->tsteps[tsID].recIDs[rindex] = recID;
                 }
               break;
@@ -376,10 +378,10 @@ extScanTimestep2(stream_t *streamptr)
   int nrecs = 0;
   for (int recID = 0; recID < nrecords; recID++)
     {
-      if (records[recID].used)
+      if (recinfo[recID].used)
         nrecs++;
       else
-        vlistDefVarTimetype(vlistID, records[recID].varID, TIME_CONSTANT);
+        vlistDefVarTimetype(vlistID, recinfo[recID].varID, TIME_CONSTANT);
     }
   streamptr->tsteps[tsID].nrecs = nrecs;
 
@@ -417,7 +419,7 @@ extScanTimestep(stream_t *streamptr)
 
   if (streamptr->tsteps[tsID].recordSize == 0)
     {
-      cdi_create_records(streamptr, tsID);
+      cdi_create_records(streamptr, tsID, true);
       record_t *records = streamptr->tsteps[tsID].records;
 
       nrecs = streamScanInitRecords(streamptr, tsID);
diff --git a/src/stream_grb.c b/src/stream_grb.c
index 9441a53e3876091937dc1881c924dd5b05afde71..80943755164a82727eee42f46a054cf339dad9c1 100644
--- a/src/stream_grb.c
+++ b/src/stream_grb.c
@@ -256,6 +256,7 @@ grbDefField(stream_t *streamptr)
   UNUSED(streamptr);
 }
 
+#ifdef HAVE_LIBGRIB
 static long
 grbScanTimestep1(stream_t *streamptr)
 {
@@ -318,6 +319,7 @@ grbScanTimestep(stream_t *streamptr)
 
   return status;
 }
+#endif
 
 #ifdef HAVE_LIBGRIB
 long
@@ -346,7 +348,6 @@ fdbInqContents(stream_t *streamptr)
 #endif
 }
 #endif
-#endif
 
 int
 fdbInqTimestep(stream_t *streamptr, int tsID)
@@ -401,32 +402,6 @@ grbInqTimestep(stream_t *streamptr, int tsID)
   return nrecs;
 }
 
-// used in CDO!!!
-void
-streamInqGRIBinfo(int streamID, int *intnum, float *fltnum, off_t *bignum)
-{
-  stream_t *streamptr = stream_to_pointer(streamID);
-
-  int filetype = streamptr->filetype;
-
-  if (filetype == CDI_FILETYPE_GRB)
-    {
-      int tsID = streamptr->curTsID;
-      int vrecID = streamptr->tsteps[tsID].curRecID;
-      int recID = streamptr->tsteps[tsID].recIDs[vrecID];
-      off_t recpos = streamptr->tsteps[tsID].records[recID].position;
-      int zip = streamptr->tsteps[tsID].records[recID].zip;
-
-      void *gribbuffer = streamptr->record->buffer;
-      size_t gribbuffersize = streamptr->record->buffersize;
-
-      if (zip > 0)
-        Error("Compressed GRIB records unsupported!");
-      else
-        grib_info_for_grads(recpos, (long) gribbuffersize, (unsigned char *) gribbuffer, intnum, fltnum, bignum);
-    }
-}
-
 int
 grbGetGridtype(int *gridID, SizeType gridsize, bool *gridIsRotated, bool *gridIsCurvilinear)
 {
@@ -497,6 +472,35 @@ grbGetGridtype(int *gridID, SizeType gridsize, bool *gridIsRotated, bool *gridIs
 
   return gridtype;
 }
+#endif
+
+// used in CDO!!!
+void
+streamInqGRIBinfo(int streamID, int *intnum, float *fltnum, off_t *bignum)
+{
+  stream_t *streamptr = stream_to_pointer(streamID);
+
+  if (streamptr->filetype == CDI_FILETYPE_GRB)
+    {
+#ifdef HAVE_LIBGRIB
+      int tsID = streamptr->curTsID;
+      int vrecID = streamptr->tsteps[tsID].curRecID;
+      int recID = streamptr->tsteps[tsID].recIDs[vrecID];
+      off_t recpos = streamptr->tsteps[tsID].records[recID].position;
+      int zip = streamptr->tsteps[tsID].records[recID].zip;
+
+      void *gribbuffer = streamptr->record->buffer;
+      size_t gribbuffersize = streamptr->record->buffersize;
+
+      if (zip > 0)
+        Error("Compressed GRIB records unsupported!");
+      else
+        grib_info_for_grads(recpos, (long) gribbuffersize, (unsigned char *) gribbuffer, intnum, fltnum, bignum);
+#else
+      Error("GRIB support unavailable!");
+#endif
+    }
+}
 
 /*
  * Local Variables:
diff --git a/src/stream_gribapi.c b/src/stream_gribapi.c
index 30c14eba1bfd748fa5a7910663f93bc2ba17ec2d..9e92946f6d3d2d808d1a16403ec785a23e9c9a16 100644
--- a/src/stream_gribapi.c
+++ b/src/stream_gribapi.c
@@ -28,12 +28,12 @@ static const var_tile_t dummy_tiles = { 0, -1, -1, -1, -1, -1 };
 
 typedef struct
 {
+  size_t gridsize;
   int param;
   int level1;
   int level2;
-  int ltype;
-  int tsteptype;
-  size_t gridsize;
+  short ltype;
+  short tsteptype;
   char name[32];
   VarScanKeys scanKeys;
   var_tile_t tiles;
@@ -710,7 +710,8 @@ gribapiAddRecord(stream_t *streamptr, int param, grib_handle *gh, size_t recsize
   int vlistID = streamptr->vlistID;
   int tsID = streamptr->curTsID;
   int recID = recordNewEntry(streamptr, tsID);
-  record_t *record = &streamptr->tsteps[tsID].records[recID];
+  recinfo_t *recinfo = &(streamptr->tsteps[tsID].recinfo[recID]);
+  record_t *record = &(streamptr->tsteps[tsID].records[recID]);
 
   int tsteptype = gribapiGetTsteptype(gh);
 
@@ -721,7 +722,7 @@ gribapiAddRecord(stream_t *streamptr, int param, grib_handle *gh, size_t recsize
   record->param = param;
   record->ilevel = level1;
   record->ilevel2 = level2;
-  record->ltype = leveltype1;
+  record->ltype = (short) leveltype1;
   record->tsteptype = (short) tsteptype;
   record->gridsize = gribapiGetGridsize(gh);
   record->scanKeys = *scanKeys;
@@ -743,7 +744,7 @@ gribapiAddRecord(stream_t *streamptr, int param, grib_handle *gh, size_t recsize
   CdiQuery *query = streamptr->query;
   if (query && cdiQueryName(query, varname) < 0)
     {
-      record->used = false;
+      recinfo->used = false;
       return;
     }
 
@@ -814,8 +815,8 @@ gribapiAddRecord(stream_t *streamptr, int param, grib_handle *gh, size_t recsize
   varAddRecord(recID, param, gridID, zaxistype, lbounds, level1, level2, level_sf, level_unit, datatype, &varID, &levelID,
                tsteptype, leveltype1, leveltype2, varname, scanKeys, tiles, &tile_index);
 
-  record->varID = (short) varID;
-  record->levelID = levelID;
+  recinfo->varID = (short) varID;
+  recinfo->levelID = levelID;
 
   varDefCompType(varID, comptype);
 
@@ -910,8 +911,8 @@ gribapiVarCompare(const compvar2_t *compVar, const record_t *record, int flag)
   compVar0.param = record->param;
   compVar0.level1 = record->ilevel;
   compVar0.level2 = record->ilevel2;
-  compVar0.ltype = record->ltype;
-  compVar0.tsteptype = record->tsteptype;
+  compVar0.ltype = (short) record->ltype;
+  compVar0.tsteptype = (short) record->tsteptype;
   compVar0.gridsize = record->gridsize;
   memcpy(compVar0.name, record->varname, sizeof(compVar->name));
 
@@ -1173,7 +1174,8 @@ fdbScanTimesteps(stream_t *streamptr)
 
       taxis = &streamptr->tsteps[tsID].taxis;
 
-      cdi_create_records(streamptr, tsID);
+      cdi_create_records(streamptr, tsID, true);
+      recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
       record_t *records = streamptr->tsteps[tsID].records;
 
       int nrecs = (tsID == 1) ? streamScanInitRecords2(streamptr) : streamScanInitRecords(streamptr, tsID);
@@ -1184,7 +1186,7 @@ fdbScanTimesteps(stream_t *streamptr)
       int rindex = 0;
       for (int recID = 0; recID < numRecords; recID++)
         {
-          records[recID].used = true;
+          recinfo[recID].used = true;
           streamptr->tsteps[tsID].recIDs[rindex] = recID;
           rindex++;
 
@@ -1209,6 +1211,7 @@ fdbScanTimesteps(stream_t *streamptr)
 }
 #endif
 
+/*
 static int
 records_cmp_varname(const void *s1, const void *s2)
 {
@@ -1216,13 +1219,14 @@ records_cmp_varname(const void *s1, const void *s2)
   return strcmp(x->varname, y->varname);
 }
 
-void
+static void
 sort_records(stream_t *streamptr)
 {
   record_t *records = streamptr->tsteps[0].records;
   size_t numRecords = (size_t) streamptr->tsteps[0].recordSize;
   qsort(records, numRecords, sizeof(records[0]), records_cmp_varname);
 }
+*/
 
 long
 gribapiScanTimestep1(stream_t *streamptr)
@@ -1373,7 +1377,7 @@ gribapiScanTimestep1(stream_t *streamptr)
   streamScanTsFixNtsteps(streamptr, recpos);
   streamScanTimeConstAdjust(streamptr, taxis);
 
-  if (streamptr->sortname) sort_records(streamptr);
+  // if (streamptr->sortname) sort_records(streamptr);
 
   return 0;
 }
@@ -1404,7 +1408,8 @@ gribapiScanTimestep2(stream_t *streamptr)
 
   fileSetPos(fileID, streamptr->tsteps[tsID].position, SEEK_SET);
 
-  cdi_create_records(streamptr, tsID);
+  cdi_create_records(streamptr, tsID, true);
+  recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
   record_t *records = streamptr->tsteps[tsID].records;
 
   int nrecords = streamScanInitRecords2(streamptr);
@@ -1492,7 +1497,7 @@ gribapiScanTimestep2(stream_t *streamptr)
             }
         }
 
-      if (records[recID].used)
+      if (recinfo[recID].used)
         {
           if (CDI_Inventory_Mode == 1)
             break;
@@ -1505,7 +1510,7 @@ gribapiScanTimestep2(stream_t *streamptr)
             }
         }
 
-      records[recID].used = true;
+      recinfo[recID].used = true;
       streamptr->tsteps[tsID].recIDs[rindex] = recID;
 
       if (CDI_Debug)
@@ -1526,7 +1531,7 @@ gribapiScanTimestep2(stream_t *streamptr)
       records[recID].position = recpos;
       records[recID].size = recsize;
 
-      int varID = records[recID].varID;
+      int varID = recinfo[recID].varID;
 
       if (tsteptype != vlistInqVarTsteptype(vlistID, varID)) vlistDefVarTsteptype(vlistID, varID, tsteptype);
 
@@ -1539,10 +1544,10 @@ gribapiScanTimestep2(stream_t *streamptr)
   int nrecs = 0;
   for (recID = 0; recID < nrecords; recID++)
     {
-      if (records[recID].used)
+      if (recinfo[recID].used)
         nrecs++;
       else
-        vlistDefVarTimetype(vlistID, records[recID].varID, TIME_CONSTANT);
+        vlistDefVarTimetype(vlistID, recinfo[recID].varID, TIME_CONSTANT);
     }
   streamptr->tsteps[tsID].nrecs = nrecs;
 
@@ -1570,7 +1575,8 @@ gribapiScanTimestep(stream_t *streamptr)
       void *gribbuffer = streamptr->record->buffer;
       size_t buffersize = streamptr->record->buffersize;
 
-      cdi_create_records(streamptr, tsID);
+      cdi_create_records(streamptr, tsID, true);
+      recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
       record_t *records = streamptr->tsteps[tsID].records;
 
       nrecs = streamScanInitRecords(streamptr, tsID);
@@ -1679,7 +1685,7 @@ gribapiScanTimestep(stream_t *streamptr)
 
           if (CDI_Inventory_Mode != 1)
             {
-              if (records[recID].used)
+              if (recinfo[recID].used)
                 {
                   if (cdiDateTime_isNE(vDateTime, vDateTime0)) break;
 
@@ -1690,7 +1696,7 @@ gribapiScanTimestep(stream_t *streamptr)
                 }
             }
 
-          records[recID].used = true;
+          recinfo[recID].used = true;
           streamptr->tsteps[tsID].recIDs[rindex] = recID;
 
           if (CDI_Debug)
@@ -1717,7 +1723,7 @@ gribapiScanTimestep(stream_t *streamptr)
       for (vrecID = 0; vrecID < nrecs; vrecID++)
         {
           recID = streamptr->tsteps[tsID].recIDs[vrecID];
-          if (!records[recID].used) break;
+          if (!recinfo[recID].used) break;
         }
 
       if (vrecID < nrecs)
@@ -2749,7 +2755,7 @@ gribapiDefLevel(int editionNumber, grib_handle *gh, int zaxisID, int levelID, in
   }
   long grib_ltype2 = (ltype != ltype2 && ltype2 != -1) ? ltype2 : grib_ltype;
 
-  void (*defLevel)(grib_handle * gh, int gcinit, long leveltype1, long leveltype2, bool hasBounds, double level, double dlevel1,
+  void (*defLevel)(grib_handle *gh, int gcinit, long leveltype1, long leveltype2, bool hasBounds, double level, double dlevel1,
                    double dlevel2)
       = (editionNumber <= 1) ? grib1DefLevel : grib2DefLevel;
 
@@ -3176,7 +3182,7 @@ convertDataScanningMode(int scanModeIN, int scanModeOUT, double *data, size_t gr
               // printf(".\n");
             }
         }  // end if (scanModeOUT==96)
-    }      // end if (scanModeIN==64)
+    }  // end if (scanModeIN==64)
 
   if (scanModeIN
       == 00)  // Scanning Mode (00 dec)  +i, -j; i direction consecutive (row-major    order West->East   & South->North )
@@ -3206,7 +3212,7 @@ convertDataScanningMode(int scanModeIN, int scanModeOUT, double *data, size_t gr
               for (size_t i = 0; i < iDim; i++) data[j + i * jDim] = dataCopy[i + jInv * iDim];  // source data has -j
             }
         }  // end if (scanModeOUT==96)
-    }      // end if (scanModeIN==00)
+    }  // end if (scanModeIN==00)
 
   if (scanModeIN
       == 96)  // Scanning Mode (00 dec)  +i, -j; i direction consecutive (row-major    order West->East   & South->North )
@@ -3239,7 +3245,7 @@ convertDataScanningMode(int scanModeIN, int scanModeOUT, double *data, size_t gr
                 data[i + jXiDim] = dataCopy[jInv + i * jDim];  // target data has -j
             }
         }  // end if (scanModeOUT==00)
-    }      // end if (scanModeIN==96)
+    }  // end if (scanModeIN==96)
 
   if (cdiDebugExt >= 100)
     {
diff --git a/src/stream_ieg.c b/src/stream_ieg.c
index 4194ed0373cba5a384ee4cbfc4515d54dbee1aa6..9caeadf6f28781c4bdeae575890e6ed60ae21418 100644
--- a/src/stream_ieg.c
+++ b/src/stream_ieg.c
@@ -44,7 +44,7 @@ ieg_read_recordSP(stream_t *streamptr, float *data, size_t *numMissVals)
 
   int vrecID = streamptr->tsteps[tsID].curRecID;
   int recID = streamptr->tsteps[tsID].recIDs[vrecID];
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
+  int varID = streamptr->tsteps[tsID].recinfo[recID].varID;
   off_t recpos = streamptr->tsteps[tsID].records[recID].position;
 
   fileSetPos(fileID, recpos, SEEK_SET);
@@ -71,7 +71,7 @@ ieg_read_recordDP(stream_t *streamptr, double *data, size_t *numMissVals)
 
   int vrecID = streamptr->tsteps[tsID].curRecID;
   int recID = streamptr->tsteps[tsID].recIDs[vrecID];
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
+  int varID = streamptr->tsteps[tsID].recinfo[recID].varID;
   off_t recpos = streamptr->tsteps[tsID].records[recID].position;
 
   fileSetPos(fileID, recpos, SEEK_SET);
@@ -521,7 +521,8 @@ iegAddRecord(stream_t *streamptr, int param, int *pdb, int *gdb, double *vct, si
   int vlistID = streamptr->vlistID;
   int tsID = streamptr->curTsID;
   int recID = recordNewEntry(streamptr, tsID);
-  record_t *record = &streamptr->tsteps[tsID].records[recID];
+  recinfo_t *recinfo = &(streamptr->tsteps[tsID].recinfo[recID]);
+  record_t *record = &(streamptr->tsteps[tsID].records[recID]);
 
   int level1, level2;
   if (IEG_P_LevelType(pdb) == IEG_LTYPE_HYBRID_LAYER)
@@ -541,7 +542,7 @@ iegAddRecord(stream_t *streamptr, int param, int *pdb, int *gdb, double *vct, si
   record->param = param;
   record->ilevel = level1;
   record->ilevel2 = level2;
-  record->ltype = IEG_P_LevelType(pdb);
+  record->ltype = (short) IEG_P_LevelType(pdb);
 
   int gridtype = (IEG_G_GridType(gdb) == 0)    ? GRID_LONLAT
                  : (IEG_G_GridType(gdb) == 10) ? GRID_PROJECTION
@@ -641,8 +642,8 @@ iegAddRecord(stream_t *streamptr, int param, int *pdb, int *gdb, double *vct, si
   varAddRecord(recID, param, gridID, leveltype, lbounds, level1, level2, 0, 0, datatype, &varID, &levelID, TSTEP_INSTANT, 0, -1,
                NULL, NULL, NULL, NULL);
 
-  record->varID = (short) varID;
-  record->levelID = levelID;
+  recinfo->varID = (short) varID;
+  recinfo->levelID = levelID;
 
   streamptr->tsteps[tsID].nallrecs++;
   streamptr->nrecs++;
@@ -791,7 +792,8 @@ iegScanTimestep2(stream_t *streamptr)
 
   fileSetPos(fileID, streamptr->tsteps[tsID].position, SEEK_SET);
 
-  cdi_create_records(streamptr, tsID);
+  cdi_create_records(streamptr, tsID, true);
+  recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
   record_t *records = streamptr->tsteps[tsID].records;
 
   int nrecords = streamScanInitRecords2(streamptr);
@@ -828,13 +830,13 @@ iegScanTimestep2(stream_t *streamptr)
         {
           if (param == records[recID].param && rlevel == records[recID].ilevel)
             {
-              if (records[recID].used)
+              if (recinfo[recID].used)
                 {
                   nextstep = true;
                 }
               else
                 {
-                  records[recID].used = true;
+                  recinfo[recID].used = true;
                   streamptr->tsteps[tsID].recIDs[rindex] = recID;
                 }
               break;
@@ -866,10 +868,10 @@ iegScanTimestep2(stream_t *streamptr)
   int nrecs = 0;
   for (int recID = 0; recID < nrecords; recID++)
     {
-      if (records[recID].used)
+      if (recinfo[recID].used)
         nrecs++;
       else
-        vlistDefVarTimetype(vlistID, records[recID].varID, TIME_CONSTANT);
+        vlistDefVarTimetype(vlistID, recinfo[recID].varID, TIME_CONSTANT);
     }
   streamptr->tsteps[tsID].nrecs = nrecs;
 
@@ -908,7 +910,7 @@ iegScanTimestep(stream_t *streamptr)
   int nrecs = 0;
   if (streamptr->tsteps[tsID].recordSize == 0)
     {
-      cdi_create_records(streamptr, tsID);
+      cdi_create_records(streamptr, tsID, true);
       record_t *records = streamptr->tsteps[tsID].records;
 
       nrecs = streamScanInitRecords(streamptr, tsID);
diff --git a/src/stream_record.c b/src/stream_record.c
index 114b0df4065a60eb624c4f4eff57a749c80807cf..5e36249758bf40be8a63772f6144608f0e7c176c 100644
--- a/src/stream_record.c
+++ b/src/stream_record.c
@@ -17,6 +17,14 @@
 #include "stream_ieg.h"
 
 void
+recinfoInitEntry(recinfo_t *recinfo)
+{
+  recinfo->varID = CDI_UNDEFID;
+  recinfo->levelID = CDI_UNDEFID;
+  recinfo->used = false;
+}
+
+static void
 recordInitEntry(record_t *record)
 {
   record->position = CDI_UNDEFID;
@@ -24,13 +32,14 @@ recordInitEntry(record_t *record)
   record->gridsize = 0;
   record->param = 0;
   record->ilevel = CDI_UNDEFID;
-  record->used = false;
   record->tsteptype = CDI_UNDEFID;
-  record->varID = CDI_UNDEFID;
-  record->levelID = CDI_UNDEFID;
-  memset(record->varname, 0, sizeof(record->varname));
+#ifdef HAVE_LIBGRIB
   varScanKeysInit(&record->scanKeys);
   memset(&record->tiles, 0, sizeof(record->tiles));
+#ifdef HAVE_LIBGRIB_API
+  memset(record->varname, 0, sizeof(record->varname));
+#endif
+#endif
 #ifdef HAVE_LIBFDB5
   record->fdbItemIndex = -1;
 #endif
@@ -40,19 +49,21 @@ int
 recordNewEntry(stream_t *streamptr, int tsID)
 {
   int recordSize = streamptr->tsteps[tsID].recordSize;
+  recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
   record_t *records = streamptr->tsteps[tsID].records;
 
   // Look for a free slot in record.
   int recordID = 0;
   if (recordSize)
     {
-      while (recordID < recordSize && records[recordID].used != CDI_UNDEFID) ++recordID;
+      while (recordID < recordSize && recinfo[recordID].used != CDI_UNDEFID) ++recordID;
     }
   else  // Create the table the first time through.
     {
       recordSize = 1;  //  <<<<----
+      recinfo = (recinfo_t *) Malloc((size_t) recordSize * sizeof(recinfo_t));
       records = (record_t *) Malloc((size_t) recordSize * sizeof(record_t));
-      for (int i = recordID; i < recordSize; i++) records[i].used = CDI_UNDEFID;
+      for (int i = recordID; i < recordSize; i++) recinfo[i].used = CDI_UNDEFID;
     }
 
   // If the table overflows, double its size.
@@ -63,16 +74,19 @@ recordNewEntry(stream_t *streamptr, int tsID)
       else if (recordSize < INT_MAX)      recordSize = INT_MAX;
       else Error("Cannot handle this many records!\n");
       // clang-format on
+      recinfo = (recinfo_t *) Realloc(recinfo, (size_t) recordSize * sizeof(recinfo_t));
       records = (record_t *) Realloc(records, (size_t) recordSize * sizeof(record_t));
 
-      for (int i = recordID; i < recordSize; i++) records[i].used = CDI_UNDEFID;
+      for (int i = recordID; i < recordSize; i++) recinfo[i].used = CDI_UNDEFID;
     }
 
+  recinfoInitEntry(&recinfo[recordID]);
   recordInitEntry(&records[recordID]);
 
-  records[recordID].used = true;
+  recinfo[recordID].used = true;
 
   streamptr->tsteps[tsID].recordSize = recordSize;
+  streamptr->tsteps[tsID].recinfo = recinfo;
   streamptr->tsteps[tsID].records = records;
 
   return recordID;
@@ -97,38 +111,48 @@ cdiInitRecord(stream_t *streamptr)
 }
 
 void
-streamInqField(int streamID, int *varID, int *levelID)
+stream_inq_field(stream_t *streamPtr, int *varID, int *levelID)
 {
   check_parg(varID);
   check_parg(levelID);
 
-  stream_t *streamptr = stream_to_pointer(streamID);
-
-  cdiDefAccesstype(streamID, TYPE_REC);
-
-  if (!streamptr->record) cdiInitRecord(streamptr);
+  stream_def_accesstype(streamPtr, TYPE_REC);
 
-  int tsID = streamptr->curTsID;
-  int rindex = streamptr->tsteps[tsID].curRecID + 1;
+  if (!streamPtr->record) cdiInitRecord(streamPtr);
 
-  if (rindex >= streamptr->tsteps[tsID].nrecs) Error("record %d not available at timestep %d", rindex + 1, tsID + 1);
+  const int tsID = streamPtr->curTsID;
+  tsteps_t *tstep = &(streamPtr->tsteps[tsID]);
 
-  int recID = streamptr->tsteps[tsID].recIDs[rindex];
+  int rindex = tstep->curRecID + 1;
+  if (rindex >= tstep->nrecs) Error("record %d not available at timestep %d", rindex + 1, tsID + 1);
 
-  if (recID == -1 || recID >= streamptr->tsteps[tsID].nallrecs) Error("Internal problem! tsID = %d recID = %d", tsID, recID);
+  int recID = tstep->recIDs[rindex];
+  if (recID == -1 || recID >= tstep->nallrecs) Error("Internal problem! tsID = %d recID = %d", tsID, recID);
 
-  *varID = streamptr->tsteps[tsID].records[recID].varID;
+  *varID = tstep->recinfo[recID].varID;
   if (*varID == -1) Error("Internal problem! varID = %d recID = %d", *varID, recID);
 
-  int lindex = streamptr->tsteps[tsID].records[recID].levelID;
+  int lindex = tstep->recinfo[recID].levelID;
+  int isub = subtypeInqActiveIndex(streamPtr->vars[*varID].subtypeID);
+  *levelID = streamPtr->vars[*varID].recordTable[isub].lindex[lindex];
+
+  if (CDI_Debug)
+    Message("streamID = %d tsID = %d, recID = %d, varID = %d, levelID = %d", streamPtr->self, tsID, recID, *varID, *levelID);
 
-  int isub = subtypeInqActiveIndex(streamptr->vars[*varID].subtypeID);
-  *levelID = streamptr->vars[*varID].recordTable[isub].lindex[lindex];
+  streamPtr->curTsID = tsID;
+  tstep->curRecID = rindex;
+}
 
-  if (CDI_Debug) Message("streamID = %d tsID = %d, recID = %d, varID = %d, levelID = %d", streamID, tsID, recID, *varID, *levelID);
+void
+pstreamInqField(void *streamPtr, int *varID, int *levelID)
+{
+  stream_inq_field((stream_t *) streamPtr, varID, levelID);
+}
 
-  streamptr->curTsID = tsID;
-  streamptr->tsteps[tsID].curRecID = rindex;
+void
+streamInqField(int streamID, int *varID, int *levelID)
+{
+  stream_inq_field(stream_to_pointer(streamID), varID, levelID);
 }
 
 /*
@@ -202,8 +226,11 @@ streamDefField(int streamID, int varID, int levelID)
 void
 streamCopyField(int streamID2, int streamID1)
 {
-  stream_t *streamptr1 = stream_to_pointer(streamID1), *streamptr2 = stream_to_pointer(streamID2);
-  int filetype1 = streamptr1->filetype, filetype2 = streamptr2->filetype, filetype = CDI_FILETYPE_UNDEF;
+  stream_t *streamptr1 = stream_to_pointer(streamID1);
+  stream_t *streamptr2 = stream_to_pointer(streamID2);
+  int filetype1 = streamptr1->filetype;
+  int filetype2 = streamptr2->filetype;
+  int filetype = CDI_FILETYPE_UNDEF;
 
   if (cdiBaseFiletype(filetype1) == cdiBaseFiletype(filetype2)) filetype = filetype2;
 
@@ -225,21 +252,21 @@ streamCopyField(int streamID2, int streamID1)
     case CDI_FILETYPE_IEG: iegCopyField(streamptr2, streamptr1); break;
 #endif
 #ifdef HAVE_LIBNETCDF
-    case CDI_FILETYPE_NETCDF: cdfCopyRecord(streamptr2, streamptr1); break;
+    case CDI_FILETYPE_NETCDF: cdfCopyField(streamptr2, streamptr1); break;
 #endif
     default: Error("%s support not compiled in!", strfiletype(filetype));
     }
 }
 
 void
-cdi_create_records(stream_t *streamptr, int tsID)
+cdi_create_records(stream_t *streamptr, int tsID, bool allocRecords)
 {
   unsigned nrecords, maxrecords;
 
   tsteps_t *sourceTstep = streamptr->tsteps;
   tsteps_t *destTstep = sourceTstep + tsID;
 
-  if (destTstep->records) return;
+  if (destTstep->recinfo) return;
 
   int vlistID = streamptr->vlistID;
 
@@ -268,7 +295,7 @@ cdi_create_records(stream_t *streamptr, int tsID)
         {
           for (size_t recID = 0; recID < maxrecords; recID++)
             {
-              int varID = sourceTstep->records[recID].varID;
+              int varID = sourceTstep->recinfo[recID].varID;
               nrecords += (varID == CDI_UNDEFID /* varID = CDI_UNDEFID for write mode !!! */
                            || vlistInqVarTimetype(vlistID, varID) != TIME_CONSTANT);
               //    printf("varID nrecords %d %d %d \n", varID, nrecords, vlistInqVarTsteptype(vlistID, varID));
@@ -285,34 +312,39 @@ cdi_create_records(stream_t *streamptr, int tsID)
     }
   //  printf("tsID, nrecords %d %d\n", tsID, nrecords);
 
-  record_t *records = (maxrecords > 0) ? (record_t *) (Malloc(maxrecords * sizeof(record_t))) : (record_t *) NULL;
+  recinfo_t *recinfo = (maxrecords > 0) ? (recinfo_t *) Malloc(maxrecords * sizeof(recinfo_t)) : (recinfo_t *) NULL;
+  record_t *records = (allocRecords && maxrecords > 0) ? (record_t *) Malloc(maxrecords * sizeof(record_t)) : (record_t *) NULL;
 
+  destTstep->recinfo = recinfo;
   destTstep->records = records;
   destTstep->recordSize = (int) maxrecords;
   destTstep->nallrecs = (int) nrecords;
 #ifdef HAVE_LIBFDB5
-  destTstep->records->fdbItemIndex = -1;
+  if (destTstep->records) destTstep->records->fdbItemIndex = -1;
 #endif
 
   if (tsID == 0)
     {
-      for (unsigned recID = 0; recID < maxrecords; recID++) recordInitEntry(&destTstep->records[recID]);
+      for (unsigned recID = 0; recID < maxrecords; recID++) recinfoInitEntry(&destTstep->recinfo[recID]);
+      if (allocRecords)
+        for (unsigned recID = 0; recID < maxrecords; recID++) recordInitEntry(&destTstep->records[recID]);
     }
-  else if (sourceTstep->records)
+  else if (sourceTstep->recinfo)
     {
-      memcpy(destTstep->records, sourceTstep->records, (size_t) maxrecords * sizeof(record_t));
+      memcpy(destTstep->recinfo, sourceTstep->recinfo, (size_t) maxrecords * sizeof(recinfo_t));
+      if (allocRecords) memcpy(destTstep->records, sourceTstep->records, (size_t) maxrecords * sizeof(record_t));
 
       for (size_t recID = 0; recID < maxrecords; recID++)
         {
-          record_t *curRecord = &sourceTstep->records[recID];
-          destTstep->records[recID].used = curRecord->used;
+          recinfo_t *curRecord = &sourceTstep->recinfo[recID];
+          destTstep->recinfo[recID].used = curRecord->used;
           if (curRecord->used != CDI_UNDEFID && curRecord->varID != -1)  // curRecord->varID = -1 for write mode !!!
             {
               if (vlistInqVarTimetype(vlistID, curRecord->varID) != TIME_CONSTANT)
                 {
-                  destTstep->records[recID].position = CDI_UNDEFID;
-                  destTstep->records[recID].size = 0;
-                  destTstep->records[recID].used = false;
+                  if (allocRecords) destTstep->records[recID].position = CDI_UNDEFID;
+                  if (allocRecords) destTstep->records[recID].size = 0;
+                  destTstep->recinfo[recID].used = false;
                 }
             }
         }
diff --git a/src/stream_scan.c b/src/stream_scan.c
index b1f4488a75c11d0ab4e836e030391f443ff89053..a4531bf581db4259ed3accf30ed60cd60ca905ff 100644
--- a/src/stream_scan.c
+++ b/src/stream_scan.c
@@ -16,10 +16,11 @@
 void
 streamScanResizeRecords1(stream_t *streamptr)
 {
-  const int nrecords = streamptr->tsteps[0].nallrecs;
+  int nrecords = streamptr->tsteps[0].nallrecs;
   if (nrecords < streamptr->tsteps[0].recordSize)
     {
       streamptr->tsteps[0].recordSize = nrecords;
+      streamptr->tsteps[0].recinfo = (recinfo_t *) Realloc(streamptr->tsteps[0].recinfo, (size_t) nrecords * sizeof(recinfo_t));
       streamptr->tsteps[0].records = (record_t *) Realloc(streamptr->tsteps[0].records, (size_t) nrecords * sizeof(record_t));
     }
 
@@ -31,7 +32,7 @@ streamScanResizeRecords1(stream_t *streamptr)
 int
 streamScanInitRecords2(stream_t *streamptr)
 {
-  const int nrecords = streamptr->tsteps[1].nallrecs;
+  int nrecords = streamptr->tsteps[1].nallrecs;
   streamptr->tsteps[1].recIDs = (int *) Malloc((size_t) nrecords * sizeof(int));
   streamptr->tsteps[1].nrecs = 0;
 
@@ -48,7 +49,7 @@ streamScanInitRecords2(stream_t *streamptr)
 int
 streamScanInitRecords(stream_t *streamptr, int tsID)
 {
-  const int nrecs = streamptr->tsteps[1].nrecs;
+  int nrecs = streamptr->tsteps[1].nrecs;
 
   streamptr->tsteps[tsID].nrecs = nrecs;
   streamptr->tsteps[tsID].recIDs = (int *) Malloc((size_t) nrecs * sizeof(int));
@@ -61,7 +62,7 @@ streamScanInitRecords(stream_t *streamptr, int tsID)
 void
 streamScanTimeConstAdjust(stream_t *streamptr, const taxis_t *taxis)
 {
-  const int vlistID = streamptr->vlistID;
+  int vlistID = streamptr->vlistID;
   if (streamptr->ntsteps == 1 && cdiDateTime_isNull(taxis->vDateTime))
     {
       streamptr->ntsteps = 0;
@@ -74,7 +75,7 @@ streamScanTsFixNtsteps(stream_t *streamptr, off_t recpos)
 {
   if (streamptr->ntsteps == -1)
     {
-      const int tsID = tstepsNewEntry(streamptr);
+      int tsID = tstepsNewEntry(streamptr);
       if (tsID != streamptr->rtsteps) Error("Internal error. tsID = %d", tsID);
 
       streamptr->tsteps[tsID - 1].next = true;
diff --git a/src/stream_srv.c b/src/stream_srv.c
index fa1beba762d18c1133569a7ef733b72232458318..61bf91f74eedb2ad94864ae5cea9f490da0869b5 100644
--- a/src/stream_srv.c
+++ b/src/stream_srv.c
@@ -43,7 +43,7 @@ srv_read_recordSP(stream_t *streamptr, float *data, size_t *numMissVals)
 
   int vrecID = streamptr->tsteps[tsID].curRecID;
   int recID = streamptr->tsteps[tsID].recIDs[vrecID];
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
+  int varID = streamptr->tsteps[tsID].recinfo[recID].varID;
   off_t recpos = streamptr->tsteps[tsID].records[recID].position;
 
   fileSetPos(fileID, recpos, SEEK_SET);
@@ -72,7 +72,7 @@ srv_read_recordDP(stream_t *streamptr, double *data, size_t *numMissVals)
 
   int vrecID = streamptr->tsteps[tsID].curRecID;
   int recID = streamptr->tsteps[tsID].recIDs[vrecID];
-  int varID = streamptr->tsteps[tsID].records[recID].varID;
+  int varID = streamptr->tsteps[tsID].recinfo[recID].varID;
   off_t recpos = streamptr->tsteps[tsID].records[recID].position;
 
   fileSetPos(fileID, recpos, SEEK_SET);
@@ -175,7 +175,8 @@ srv_add_record(stream_t *streamptr, int param, int level, size_t xsize, size_t y
   int vlistID = streamptr->vlistID;
   int tsID = streamptr->curTsID;
   int recID = recordNewEntry(streamptr, tsID);
-  record_t *record = &streamptr->tsteps[tsID].records[recID];
+  recinfo_t *recinfo = &(streamptr->tsteps[tsID].recinfo[recID]);
+  record_t *record = &(streamptr->tsteps[tsID].records[recID]);
 
   record->size = recsize;
   record->position = position;
@@ -203,8 +204,8 @@ srv_add_record(stream_t *streamptr, int param, int level, size_t xsize, size_t y
                NULL, NULL);
 
   xassert(varID <= SHRT_MAX && levelID <= SHRT_MAX);
-  record->varID = (short) varID;
-  record->levelID = levelID;
+  recinfo->varID = (short) varID;
+  recinfo->levelID = levelID;
 
   streamptr->tsteps[tsID].nallrecs++;
   streamptr->nrecs++;
@@ -313,7 +314,8 @@ srvScanTimestep2(stream_t *streamptr)
 
   fileSetPos(fileID, streamptr->tsteps[tsID].position, SEEK_SET);
 
-  cdi_create_records(streamptr, tsID);
+  cdi_create_records(streamptr, tsID, true);
+  recinfo_t *recinfo = streamptr->tsteps[tsID].recinfo;
   record_t *records = streamptr->tsteps[tsID].records;
 
   int nrecords = streamScanInitRecords2(streamptr);
@@ -349,13 +351,13 @@ srvScanTimestep2(stream_t *streamptr)
         {
           if (param == records[recID].param && rlevel == records[recID].ilevel)
             {
-              if (records[recID].used)
+              if (recinfo[recID].used)
                 {
                   nextstep = true;
                 }
               else
                 {
-                  records[recID].used = true;
+                  recinfo[recID].used = true;
                   streamptr->tsteps[tsID].recIDs[rindex] = recID;
                 }
               break;
@@ -385,10 +387,10 @@ srvScanTimestep2(stream_t *streamptr)
   int nrecs = 0;
   for (int recID = 0; recID < nrecords; recID++)
     {
-      if (records[recID].used)
+      if (recinfo[recID].used)
         nrecs++;
       else
-        vlistDefVarTimetype(vlistID, records[recID].varID, TIME_CONSTANT);
+        vlistDefVarTimetype(vlistID, recinfo[recID].varID, TIME_CONSTANT);
     }
   streamptr->tsteps[tsID].nrecs = nrecs;
 
@@ -426,7 +428,7 @@ srvScanTimestep(stream_t *streamptr)
 
   if (streamptr->tsteps[tsID].recordSize == 0)
     {
-      cdi_create_records(streamptr, tsID);
+      cdi_create_records(streamptr, tsID, true);
       record_t *records = streamptr->tsteps[tsID].records;
 
       nrecs = streamScanInitRecords(streamptr, tsID);
diff --git a/src/taxis.c b/src/taxis.c
index 5acf45b5715e6dba9f64f41f1473927f3a56ddfd..a252db8b187e6d64c52b75b43383d3f70fe18b07 100644
--- a/src/taxis.c
+++ b/src/taxis.c
@@ -102,7 +102,6 @@ void
 ptaxisInit(taxis_t *taxisptr)
 {
   taxisptr->self = CDI_UNDEFID;
-  taxisptr->datatype = CDI_DATATYPE_FLT64;
   taxisptr->type = DefaultTimeType;
   taxisptr->calendar = CDI_Default_Calendar;
   taxisptr->unit = DefaultTimeUnit;
@@ -120,6 +119,9 @@ ptaxisInit(taxis_t *taxisptr)
   taxisptr->name = NULL;
   taxisptr->longname = NULL;
   taxisptr->units = NULL;
+
+  cdiInitKeys(&taxisptr->keys);
+  cdiDefVarKeyInt(&taxisptr->keys, CDI_KEY_DATATYPE, CDI_DATATYPE_FLT64);
 }
 
 static taxis_t *
@@ -235,7 +237,7 @@ taxisDefType(int taxisID, int taxistype)
   if (taxisptr->type != taxistype)
     {
       taxisptr->type = taxistype;
-      taxisptr->datatype = CDI_DATATYPE_FLT64;
+      cdiDefVarKeyInt(&taxisptr->keys, CDI_KEY_DATATYPE, CDI_DATATYPE_FLT64);
       delete_refcount_string(taxisptr->units);
       taxisptr->units = NULL;
       reshSetStatus(taxisID, &taxisOps, RESH_DESYNC_IN_USE);
@@ -840,7 +842,7 @@ taxisInqNumavg(int taxisID)
 }
 
 taxis_t *
-taxisPtr(int taxisID)
+taxis_to_pointer(int taxisID)
 {
   taxis_t *taxisptr = (taxis_t *) reshGetVal(taxisID, &taxisOps);
   return taxisptr;
@@ -849,7 +851,7 @@ taxisPtr(int taxisID)
 void
 ptaxisDefDatatype(taxis_t *taxisptr, int datatype)
 {
-  taxisptr->datatype = datatype;
+  cdiDefVarKeyInt(&taxisptr->keys, CDI_KEY_DATATYPE, datatype);
 }
 
 void
@@ -1349,7 +1351,6 @@ ptaxisCopy(taxis_t *dest, taxis_t *source)
   reshLock();
 
   // memcpy(dest, source, sizeof(taxis_t));
-  dest->datatype = source->datatype;
   dest->type = source->type;
   dest->calendar = source->calendar;
   dest->unit = source->unit;
@@ -1374,6 +1375,9 @@ ptaxisCopy(taxis_t *dest, taxis_t *source)
   dest->units = dup_refcount_string(source->units);
   if (dest->self != CDI_UNDEFID) reshSetStatus(dest->self, &taxisOps, RESH_DESYNC_IN_USE);
 
+  cdiInitKeys(&dest->keys);
+  cdiCopyVarKeys(&source->keys, &dest->keys);
+
   reshUnlock();
 }
 
@@ -1480,6 +1484,7 @@ taxisGetPackSize(void *p, void *context)
                        + (taxisptr->longname ? serializeGetSize((int) strlen(taxisptr->longname), CDI_DATATYPE_TXT, context) : 0)
                        + (taxisptr->units ? serializeGetSize((int) strlen(taxisptr->units), CDI_DATATYPE_TXT, context) : 0)
                        + serializeGetSize(1, CDI_DATATYPE_UINT32, context);
+  packBufferSize += serializeKeysGetPackSize(&taxisptr->keys, context);
   return packBufferSize;
 }
 
@@ -1546,6 +1551,8 @@ taxisUnpack(char *unpackBuffer, int unpackBufferSize, int *unpackBufferPos, int
       taxisP->units = units;
     }
 
+  serializeKeysUnpack(unpackBuffer, unpackBufferSize, unpackBufferPos, &taxisP->keys, context);
+
   reshSetStatus(taxisP->self, &taxisOps, reshGetStatus(taxisP->self, &taxisOps) & ~RESH_SYNC_BIT);
 #undef adaptKey
 
@@ -1596,6 +1603,8 @@ taxisPack(void *voidP, void *packBuffer, int packBufferSize, int *packBufferPos,
   if (taxisP->longname)
     serializePack(taxisP->longname, lnameLen, CDI_DATATYPE_TXT, packBuffer, packBufferSize, packBufferPos, context);
   if (taxisP->units) serializePack(taxisP->units, unitsLen, CDI_DATATYPE_TXT, packBuffer, packBufferSize, packBufferPos, context);
+
+  serializeKeysPack(&taxisP->keys, packBuffer, packBufferSize, packBufferPos, context);
 }
 
 /*
diff --git a/src/taxis.h b/src/taxis.h
index b7eac28932fc67368597f4fc030714c3782325e6..5888b513d4ba6715044d2d936faae5638f780dd4 100644
--- a/src/taxis.h
+++ b/src/taxis.h
@@ -4,6 +4,8 @@
 #include <stdbool.h>
 #include "cdi.h"
 
+#include "cdi_key.h"
+
 #ifndef RESOURCE_HANDLE_H
 #include "resource_handle.h"
 #endif
@@ -11,8 +13,7 @@
 typedef struct
 {
   int self;
-  int datatype;  // datatype
-  int type;      // time type
+  int type;  // time type
   int calendar;
   int unit;  // time units
   int numavg;
@@ -29,6 +30,7 @@ typedef struct
   char *units;
   bool climatology;
   bool hasBounds;
+  cdi_keys_t keys;
 } taxis_t;
 
 //      taxisInqSdatetime: Get the start date/time
@@ -36,7 +38,7 @@ CdiDateTime taxisInqSdatetime(int taxisID);
 
 void ptaxisInit(taxis_t *taxis);
 void ptaxisCopy(taxis_t *dest, taxis_t *source);
-taxis_t *taxisPtr(int taxisID);
+taxis_t *taxis_to_pointer(int taxisID);
 void cdi_set_forecast_period(double timevalue, taxis_t *taxis);
 CdiDateTime cdi_decode_timeval(double timevalue, const taxis_t *taxis);
 double cdi_encode_timeval(CdiDateTime datetime, taxis_t *taxis);
diff --git a/src/tsteps.c b/src/tsteps.c
index 7a22bd24830f69e86993eb41c1e537f8d98a5cc7..d2a6da608ef3a7387a63aef2c74d902e3fd7640e 100644
--- a/src/tsteps.c
+++ b/src/tsteps.c
@@ -20,6 +20,7 @@ static void
 tstepsInitEntry(tsteps_t *tstep)
 {
   tstep->recIDs = NULL;
+  tstep->recinfo = NULL;
   tstep->records = NULL;
   tstep->recordSize = 0;
   tstep->nrecs = 0;
@@ -35,7 +36,7 @@ tstepsInitEntry(tsteps_t *tstep)
 int
 tstepsNewEntry(stream_t *streamptr)
 {
-  const int tsID = streamptr->tstepsNextID++;
+  int tsID = streamptr->tstepsNextID++;
   int tstepsTableSize = streamptr->tstepsTableSize;
   tsteps_t *tstepsTable = streamptr->tsteps;
 
diff --git a/src/util.c b/src/util.c
index 7eada35c9d6a99e63190757c9da6c3011af3e68e..df97d6a0cc6fdafd74a53c62f8848022d6c3c771 100644
--- a/src/util.c
+++ b/src/util.c
@@ -100,6 +100,7 @@ cdiUnescapeSpaces(const char *string, const char **outStringEnd)
   return result;
 }
 
+#ifndef _WIN32
 #if defined(HAVE_DECL_UUID_GENERATE) && defined(HAVE_UUID_UUID_H)
 #ifdef HAVE_SYS_TIME_H
 #include <sys/time.h>
@@ -252,6 +253,7 @@ cdiCreateUUID(unsigned char *uuid)
 #endif
 }
 #endif
+#endif
 
 /*
  * Local Variables:
diff --git a/src/vlist.c b/src/vlist.c
index be99645856a386166af6443eb295fb48b9dd6be2..a6bf4888ee0e7ebcea82061e0bcf35a52c14eb74 100644
--- a/src/vlist.c
+++ b/src/vlist.c
@@ -44,6 +44,19 @@ static bool vlistIsInitialized = false;
   if (!vlistIsInitialized) vlist_initialize()
 #endif
 
+void *
+stream_get_pointer(int streamID)
+{
+  return stream_to_pointer(streamID);
+}
+
+void *
+stream_get_vlist_pointer(int streamID)
+{
+  stream_t *streamPtr = stream_to_pointer(streamID);
+  return vlist_to_pointer(streamPtr->vlistID);
+}
+
 static int
 vlist_compare(vlist_t *a, vlist_t *b)
 {
@@ -1000,10 +1013,15 @@ int
 vlistNumGrids(int vlistID)
 {
   vlist_t *vlistptr = vlist_to_pointer(vlistID);
-
   return vlistptr->ngrids;
 }
 
+int
+vlistNgrids(int vlistID)
+{
+  return vlistNumGrids(vlistID);
+}
+
 /*
 @Function  vlistNumZaxis
 @Title     Number of zaxis in a variable list
@@ -1024,15 +1042,19 @@ int
 vlistNumZaxis(int vlistID)
 {
   vlist_t *vlistptr = vlist_to_pointer(vlistID);
-
   return vlistptr->nzaxis;
 }
 
+int
+vlistNzaxis(int vlistID)
+{
+  return vlistNumZaxis(vlistID);
+}
+
 int
 vlistNsubtypes(int vlistID)
 {
   vlist_t *vlistptr = vlist_to_pointer(vlistID);
-
   return vlistptr->nsubtypes;
 }
 
@@ -1053,7 +1075,6 @@ int
 vlistNtsteps(int vlistID)
 {
   vlist_t *vlistptr = vlist_to_pointer(vlistID);
-
   return (int) vlistptr->ntsteps;
 }
 
@@ -1329,6 +1350,22 @@ vlistGridIndex(int vlistID, int gridID)
   return index;
 }
 
+static void
+delete_chunks(int vlistID, int varID)
+{
+  int chunkKeys[4] = { CDI_KEY_CHUNKSIZE_DIMX, CDI_KEY_CHUNKSIZE_DIMY, CDI_KEY_CHUNKSIZE_DIMZ, CDI_KEY_CHUNKSIZE_DIMT };
+  for (int i = 0; i < 4; ++i)
+    {
+      int chunkSize = 0;
+      cdiInqKeyInt(vlistID, varID, chunkKeys[i], &chunkSize);
+      if (chunkSize != 0) cdiDeleteKey(vlistID, varID, chunkKeys[i]);
+    }
+
+  int chunkSize = 0;
+  cdiInqKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE, &chunkSize);
+  if (chunkSize > 0) cdiDeleteKey(vlistID, varID, CDI_KEY_CHUNKSIZE);
+}
+
 void
 vlistChangeGridIndex(int vlistID, int index, int gridID)
 {
@@ -1344,9 +1381,7 @@ vlistChangeGridIndex(int vlistID, int index, int gridID)
         if (vlistptr->vars[varID].gridID == gridIDold)
           {
             vlistptr->vars[varID].gridID = gridID;
-            int chunkSize = 0;
-            cdiInqKeyInt(vlistID, varID, CDI_KEY_CHUNKSIZE, &chunkSize);
-            if (chunkSize > 0) cdiDeleteKey(vlistID, varID, CDI_KEY_CHUNKSIZE);
+            delete_chunks(vlistID, varID);
             if (gridInqXsize(gridIDold) == 0 && gridInqXsize(gridID) > 0 && vlistInqVarXYZ(vlistID, varID) != 0)
               vlistDefVarXYZ(vlistID, varID, 0);
           }
@@ -1372,7 +1407,11 @@ vlistChangeGrid(int vlistID, int gridID1, int gridID2)
         }
       int nvars = vlistptr->nvars;
       for (int varID = 0; varID < nvars; varID++)
-        if (vlistptr->vars[varID].gridID == gridID1) vlistptr->vars[varID].gridID = gridID2;
+        if (vlistptr->vars[varID].gridID == gridID1)
+          {
+            vlistptr->vars[varID].gridID = gridID2;
+            delete_chunks(vlistID, varID);
+          }
       reshSetStatus(vlistID, &vlistOps, RESH_DESYNC_IN_USE);
     }
 }
diff --git a/src/vlist_var.c b/src/vlist_var.c
index 9965f49e83622724117a943d8d8ccaa9fbde2bb2..4174a111f189514a4e8f5739beb5d6533d251bec 100644
--- a/src/vlist_var.c
+++ b/src/vlist_var.c
@@ -1077,16 +1077,28 @@ vlistDefFlag(int vlistID, int varID, int levID, int flag)
 }
 
 int
-vlistInqFlag(int vlistID, int varID, int levID)
+vlist_inq_flag(vlist_t *vlistPtr, int varID, int levelID)
 {
-  var_t *varptr = vlistptr_get_varptr(__func__, vlist_to_pointer(vlistID), varID);
+  var_t *varPtr = vlistptr_get_varptr(__func__, vlistPtr, varID);
 
-  if (varptr->levinfo) return varptr->levinfo[levID].flag;
+  if (varPtr->levinfo) return varPtr->levinfo[levelID].flag;
 
-  levinfo_t li = DEFAULT_LEVINFO(levID);
+  levinfo_t li = DEFAULT_LEVINFO(levelID);
   return li.flag;
 }
 
+int
+pvlistInqFlag(void *vlistPtr, int varID, int levelID)
+{
+  return vlist_inq_flag((vlist_t *) vlistPtr, varID, levelID);
+}
+
+int
+vlistInqFlag(int vlistID, int varID, int levelID)
+{
+  return vlist_inq_flag(vlist_to_pointer(vlistID), varID, levelID);
+}
+
 int
 vlistFindVar(int vlistID, int fvarID)
 {
diff --git a/src/zaxis.c b/src/zaxis.c
index a026f71361df46dfe756c1cffa89f54fb4ae07fb..447defcdffb6b1465a7312f073cd1715d630a264 100644
--- a/src/zaxis.c
+++ b/src/zaxis.c
@@ -1139,7 +1139,7 @@ static inline void
 zaxisCopyKeyStr(zaxis_t *zaxisptr1, zaxis_t *zaxisptr2, int key)
 {
   cdi_key_t *keyp = find_key(&zaxisptr1->keys, key);
-  if (keyp && keyp->type == KEY_BYTES)
+  if (keyp && keyp->type == KeyBytes)
     cdiDefVarKeyBytes(&zaxisptr2->keys, key, (const unsigned char *) keyp->v.s, (int) keyp->length);
 }
 
diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt
new file mode 100644
index 0000000000000000000000000000000000000000..2e02cf49b4dee098c5d3cc809d18159f47944de3
--- /dev/null
+++ b/tests/CMakeLists.txt
@@ -0,0 +1,43 @@
+
+add_executable(calendar_test1.run calendar_test1.c)
+add_executable(cksum_read.run cksum_read.c )
+add_executable(cksum_verify.run cksum_verify.c)
+add_executable(cksum_write.run cksum_write.c)
+add_executable(cksum_write_chunk.run cksum_write_chunk.c)
+add_executable(deco2d_model.run deco2d_model.c)
+add_executable(ensure_array_size.run ensure_array_size.c)
+add_executable(pio_write.run pio_write.c)
+add_executable(pio_write_setup_grid.run pio_write_setup_grid.c)
+add_executable(simple_model.run simple_model.c)
+add_executable(simple_model_helper.run simple_model_helper.c)
+add_executable(stream_cksum.run stream_cksum.c)
+add_executable(test_byteswap.run test_byteswap.c)
+add_executable(test_cdf_read.run test_cdf_read.c)
+add_executable(test_cdf_write.run test_cdf_write.c)
+add_executable(test_grib.run test_grib.c)
+add_executable(test_month_adjust.run test_month_adjust.c)
+add_executable(test_resource_copy.run test_resource_copy.c)
+add_executable(test_table.run test_table.c)
+
+get_property(current_targets DIRECTORY ${dir} PROPERTY BUILDSYSTEM_TARGETS)
+list(APPEND test_lib_src
+  var_cksum.c
+  ensure_array_size.c
+  stream_cksum.c
+  simple_model.c
+  simple_model_helper.c
+  pio_write_setup_grid.c
+)
+
+add_library(test_lib ${test_lib_src})
+target_include_directories(test_lib PRIVATE ../src/)
+target_link_libraries(test_lib PRIVATE cdilib ${cdi_linked_libs})
+
+foreach(target ${current_targets})
+  target_include_directories(${target} PRIVATE ../src/)
+  target_link_libraries(${target} PRIVATE cdilib test_lib ${cdi_linked_libs})
+  add_test(NAME ${target} COMMAND ${target})
+  if(labels_${target})
+    set_property(TEST ${target} PROPERTY LABELS labels_${target})
+  endif()
+endforeach()
diff --git a/tests/Makefile.am b/tests/Makefile.am
index d530636e10f01568e32e65549577637f058d3f70..701e1f520f770f9ff03ba40a0483491176c98200 100644
--- a/tests/Makefile.am
+++ b/tests/Makefile.am
@@ -1,5 +1,7 @@
 TEST_EXTENSIONS = @EXEEXT@ .run
 
+EXTRA_DIST = CMakeLists.txt
+
 # Serial tests:
 TESTS = \
   calendar_test1 \
diff --git a/tests/test_resource_copy.c b/tests/test_resource_copy.c
index ce789c30dabe9dfd5fbf8f62c897b1481afd9b83..ffd2c7793e01a36f2764ec7b56c31b66bdd1d596 100644
--- a/tests/test_resource_copy.c
+++ b/tests/test_resource_copy.c
@@ -63,7 +63,7 @@ defineGrid(void)
   cdiDefKeyString(gridID, CDI_XAXIS, CDI_KEY_UNITS, "myXunits");
   cdiDefKeyString(gridID, CDI_YAXIS, CDI_KEY_UNITS, "myYunits");
 
-  gridDefDatatype(gridID, DOUBLE_PRECISION);
+  cdiDefKeyInt(gridID, CDI_GLOBAL, CDI_KEY_DATATYPE, DOUBLE_PRECISION);
   gridDefTrunc(gridID, 1);
   gridDefParamGME(gridID, 2, 3, 4, 5);
 
@@ -164,7 +164,7 @@ defineVlist(int gridID, int zaxisID, int taxisID)
   cdiDefAttTxt(vlistID, varID2, "txt demo", 6, "banana");
   vlistDefTaxis(vlistID, taxisID);
   int vlistID2 = vlistDuplicate(vlistID);
-  return (struct idPair){ vlistID, vlistID2 };
+  return (struct idPair) { vlistID, vlistID2 };
 }
 
 static int