AM_CPPFLAGS += -DWITHOUT_LZO
 endif
 
+if WITHOUT_ZSTD
+AM_CPPFLAGS += -DWITHOUT_ZSTD
+endif
+
 if WITH_SELINUX
 AM_CPPFLAGS += -DWITH_SELINUX
 endif
 
 need_uuid="no"
 need_zlib="no"
 need_lzo="no"
+need_zstd="no"
 need_xattr="no"
 need_cmocka="no"
 need_selinux="no"
        need_xattr="yes"
        need_zlib="yes"
        need_lzo="yes"
+       need_zstd="yes"
        need_openssl="yes"
 ])
 
        *) AC_MSG_ERROR([bad value ${withval} for --without-lzo]) ;;
        esac])
 
+AC_ARG_WITH([zstd],
+       [AS_HELP_STRING([--without-zstd], [Disable support for ZSTD compression])],
+       [case "${withval}" in
+       yes) ;;
+       no) need_zstd="no" ;;
+       *) AC_MSG_ERROR([bad value ${withval} for --without-zstd]) ;;
+       esac])
+
 AC_ARG_WITH([selinux],
        [AS_HELP_STRING([--with-selinux],
                [Enable support for selinux extended attributes])],
 uuid_missing="no"
 zlib_missing="no"
 lzo_missing="no"
+zstd_missing="no"
 xattr_missing="no"
 cmocka_missing="no"
 selinux_missing="no"
        )
 fi
 
+if test "x$need_zstd" = "xyes"; then
+       PKG_CHECK_MODULES([ZSTD], [libzstd],, zstd_missing="yes")
+fi
+
 if test "x$need_xattr" = "xyes"; then
        AC_CHECK_HEADERS([sys/xattr.h], [], [xattr_missing="yes"])
        AC_CHECK_HEADERS([sys/acl.h], [], [xattr_missing="yes"])
        dep_missing="yes"
 fi
 
+if test "x$zstd_missing" = "xyes"; then
+       AC_MSG_WARN([cannot find ZSTD library required for mkfs program])
+       AC_MSG_NOTICE([mtd-utils can optionally be built without mkfs.ubifs])
+       AC_MSG_NOTICE([mtd-utils can optionally be built without ZSTD support])
+       dep_missing="yes"
+fi
+
 if test "x$xattr_missing" = "xyes"; then
        AC_MSG_WARN([cannot find headers for extended attributes])
        AC_MSG_WARN([disabling XATTR support])
 ##### generate output #####
 
 AM_CONDITIONAL([WITHOUT_LZO], [test "x$need_lzo" != "xyes"])
+AM_CONDITIONAL([WITHOUT_ZSTD], [test "x$need_zstd" != "xyes"])
 AM_CONDITIONAL([WITHOUT_XATTR], [test "x$need_xattr" != "xyes"])
 AM_CONDITIONAL([WITH_SELINUX], [test "x$need_selinux" == "xyes"])
 AM_CONDITIONAL([WITH_CRYPTO], [test "x$need_openssl" == "xyes"])
 
  * UBIFS_COMPR_NONE: no compression
  * UBIFS_COMPR_LZO: LZO compression
  * UBIFS_COMPR_ZLIB: ZLIB compression
+ * UBIFS_COMPR_ZSTD: ZSTD compression
  * UBIFS_COMPR_TYPES_CNT: count of supported compression types
  */
 enum {
        UBIFS_COMPR_NONE,
        UBIFS_COMPR_LZO,
        UBIFS_COMPR_ZLIB,
+       UBIFS_COMPR_ZSTD,
        UBIFS_COMPR_TYPES_CNT,
 };
 
 
                ubifs-utils/mkfs.ubifs/fscrypt.c
 endif
 
-mkfs_ubifs_LDADD = libmtd.a libubi.a $(ZLIB_LIBS) $(LZO_LIBS) $(UUID_LIBS) $(LIBSELINUX_LIBS) $(OPENSSL_LIBS) -lm
-mkfs_ubifs_CPPFLAGS = $(AM_CPPFLAGS) $(ZLIB_CFLAGS) $(LZO_CFLAGS) $(UUID_CFLAGS) $(LIBSELINUX_CFLAGS)\
+mkfs_ubifs_LDADD = libmtd.a libubi.a $(ZLIB_LIBS) $(LZO_LIBS) $(ZSTD_LIBS) $(UUID_LIBS) $(LIBSELINUX_LIBS) $(OPENSSL_LIBS) -lm
+mkfs_ubifs_CPPFLAGS = $(AM_CPPFLAGS) $(ZLIB_CFLAGS) $(LZO_CFLAGS) $(ZSTD_CFLAGS) $(UUID_CFLAGS) $(LIBSELINUX_CFLAGS)\
        -I$(top_srcdir)/ubi-utils/include -I$(top_srcdir)/ubifs-utils/mkfs.ubifs/
 
 UBIFS_BINS = \
 
 #include <lzo/lzo1x.h>
 #endif
 #include <linux/types.h>
+#ifndef WITHOUT_ZSTSD
+#include <zstd.h>
+#endif
 
 #define crc32 __zlib_crc32
 #include <zlib.h>
 }
 #endif
 
+#ifndef WITHOUT_ZSTD
+static ZSTD_CCtx *zctx;
+
+static int zstd_compress(void *in_buf, size_t in_len, void *out_buf,
+                        size_t *out_len)
+{
+       size_t ret;
+
+       ret = ZSTD_compressCCtx(zctx, out_buf, *out_len, in_buf, in_len,
+                               ZSTD_CLEVEL_DEFAULT);
+       if (ZSTD_isError(ret)) {
+               errcnt += 1;
+               return -1;
+       }
+       *out_len = ret;
+       return 0;
+}
+#endif
+
 static int no_compress(void *in_buf, size_t in_len, void *out_buf,
                       size_t *out_len)
 {
                case MKFS_UBIFS_COMPR_ZLIB:
                        ret = zlib_deflate(in_buf, in_len, out_buf, out_len);
                        break;
+#ifndef WITHOUT_ZSTD
+               case MKFS_UBIFS_COMPR_ZSTD:
+                       ret = zstd_compress(in_buf, in_len, out_buf, out_len);
+                       break;
+#endif
                case MKFS_UBIFS_COMPR_NONE:
                        ret = 1;
                        break;
 #endif
 
        zlib_buf = malloc(UBIFS_BLOCK_SIZE * WORST_COMPR_FACTOR);
-       if (!zlib_buf) {
-               free(lzo_mem);
-               return -1;
-       }
+       if (!zlib_buf)
+               goto err;
+
+#ifndef WITHOUT_ZSTD
+       zctx = ZSTD_createCCtx();
+       if (!zctx)
+               goto err;
+#endif
 
        return 0;
+err:
+       free(zlib_buf);
+       free(lzo_mem);
+       return -1;
 }
 
 void destroy_compression(void)
 {
        free(zlib_buf);
        free(lzo_mem);
+#ifndef WITHOUT_ZSTD
+       ZSTD_freeCCtx(zctx);
+#endif
        if (errcnt)
                fprintf(stderr, "%llu compression errors occurred\n", errcnt);
 }
 
        MKFS_UBIFS_COMPR_NONE,
        MKFS_UBIFS_COMPR_LZO,
        MKFS_UBIFS_COMPR_ZLIB,
+       MKFS_UBIFS_COMPR_ZSTD,
 };
 
 int compress_data(void *in_buf, size_t in_len, void *out_buf, size_t *out_len,
 
 #include <selinux/label.h>
 #endif
 
+#ifndef WITHOUT_ZSTD
+#include <zstd.h>
+#endif
+
 #include "crypto.h"
 #include "fscrypt.h"
 
 "-o, --output=FILE        output to FILE\n"
 "-j, --jrn-size=SIZE      journal size\n"
 "-R, --reserved=SIZE      how much space should be reserved for the super-user\n"
-"-x, --compr=TYPE         compression type - \"lzo\", \"favor_lzo\", \"zlib\" or\n"
-"                         \"none\" (default: \"lzo\")\n"
+"-x, --compr=TYPE         compression type - \"lzo\", \"favor_lzo\", \"zlib\"\n"
+"                         \"zstd\" or \"none\" (default: \"lzo\")\n"
 "-X, --favor-percent      may only be used with favor LZO compression and defines\n"
 "                         how many percent better zlib should compress to make\n"
 "                         mkfs.ubifs use zlib instead of LZO (default 20%)\n"
                                c->default_compr = UBIFS_COMPR_NONE;
                        else if (strcmp(optarg, "zlib") == 0)
                                c->default_compr = UBIFS_COMPR_ZLIB;
+#ifndef WITHOUT_ZSTD
+                       else if (strcmp(optarg, "zstd") == 0)
+                               c->default_compr = UBIFS_COMPR_ZSTD;
+#endif
 #ifndef WITHOUT_LZO
                        else if (strcmp(optarg, "favor_lzo") == 0) {
                                c->default_compr = UBIFS_COMPR_LZO;
 
 #if MKFS_UBIFS_COMPR_ZLIB != UBIFS_COMPR_ZLIB
 #error MKFS_UBIFS_COMPR_ZLIB != UBIFS_COMPR_ZLIB
 #endif
+#if MKFS_UBIFS_COMPR_ZSTD != UBIFS_COMPR_ZSTD
+#error MKFS_UBIFS_COMPR_ZSTD != UBIFS_COMPR_ZSTD
+#endif
 
 extern int verbose;
 extern int debug_level;