Compare commits
	
		
			3 Commits 
		
	
	
		
			e30be08198
			...
			c3cd98bf82
		
	
	| Author | SHA1 | Date | 
|---|---|---|
| 
							
							
								 | 
						c3cd98bf82 | |
| 
							
							
								 | 
						23eb2f89a8 | |
| 
							
							
								 | 
						233c48673f | 
											
												Binary file not shown.
											
										
									
								| 
						 | 
					@ -1,6 +1,6 @@
 | 
				
			||||||
cmake_minimum_required(VERSION 3.6)
 | 
					cmake_minimum_required(VERSION 3.6)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
project(hv VERSION 1.3.2)
 | 
					project(hv VERSION 1.3.3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
option(BUILD_SHARED "build shared library" ON)
 | 
					option(BUILD_SHARED "build shared library" ON)
 | 
				
			||||||
option(BUILD_STATIC "build static library" ON)
 | 
					option(BUILD_STATIC "build static library" ON)
 | 
				
			||||||
| 
						 | 
					@ -29,7 +29,7 @@ option(WITH_MBEDTLS "with mbedtls library" OFF)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
option(WITH_KCP "compile event/kcp" OFF)
 | 
					option(WITH_KCP "compile event/kcp" OFF)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if(WIN32)
 | 
					if(WIN32 OR MINGW)
 | 
				
			||||||
    option(WITH_WEPOLL "compile event/wepoll -> use iocp" ON)
 | 
					    option(WITH_WEPOLL "compile event/wepoll -> use iocp" ON)
 | 
				
			||||||
    option(ENABLE_WINDUMP "Windows MiniDumpWriteDump" OFF)
 | 
					    option(ENABLE_WINDUMP "Windows MiniDumpWriteDump" OFF)
 | 
				
			||||||
    option(BUILD_FOR_MT "build for /MT" OFF)
 | 
					    option(BUILD_FOR_MT "build for /MT" OFF)
 | 
				
			||||||
| 
						 | 
					@ -87,6 +87,14 @@ check_function("socketpair" "sys/socket.h")
 | 
				
			||||||
check_function("eventfd" "sys/eventfd.h")
 | 
					check_function("eventfd" "sys/eventfd.h")
 | 
				
			||||||
check_function("setproctitle" "unistd.h")
 | 
					check_function("setproctitle" "unistd.h")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if (NOT HAVE_CLOCK_GETTIME)
 | 
				
			||||||
 | 
					    include(CheckLibraryExists)
 | 
				
			||||||
 | 
					    check_library_exists(rt clock_gettime "" HAVE_CLOCK_GETTIME_IN_RT)
 | 
				
			||||||
 | 
					    if (HAVE_CLOCK_GETTIME_IN_RT)
 | 
				
			||||||
 | 
					        set(HAVE_CLOCK_GETTIME ${HAVE_CLOCK_GETTIME_IN_RT})
 | 
				
			||||||
 | 
					    endif()
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/hconfig.h.in ${CMAKE_CURRENT_SOURCE_DIR}/hconfig.h)
 | 
					configure_file(${CMAKE_CURRENT_SOURCE_DIR}/hconfig.h.in ${CMAKE_CURRENT_SOURCE_DIR}/hconfig.h)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# see Makefile.in
 | 
					# see Makefile.in
 | 
				
			||||||
| 
						 | 
					@ -127,7 +135,7 @@ endif()
 | 
				
			||||||
if(WITH_CURL)
 | 
					if(WITH_CURL)
 | 
				
			||||||
    add_definitions(-DWITH_CURL)
 | 
					    add_definitions(-DWITH_CURL)
 | 
				
			||||||
    set(LIBS ${LIBS} curl)
 | 
					    set(LIBS ${LIBS} curl)
 | 
				
			||||||
    if(WIN32)
 | 
					    if(WIN32 OR MINGW)
 | 
				
			||||||
        set(LIBS ${LIBS} wldap32 advapi32 crypt32)
 | 
					        set(LIBS ${LIBS} wldap32 advapi32 crypt32)
 | 
				
			||||||
    endif()
 | 
					    endif()
 | 
				
			||||||
endif()
 | 
					endif()
 | 
				
			||||||
| 
						 | 
					@ -157,7 +165,7 @@ if(WITH_MBEDTLS)
 | 
				
			||||||
    set(LIBS ${LIBS} mbedtls mbedx509 mbedcrypto)
 | 
					    set(LIBS ${LIBS} mbedtls mbedx509 mbedcrypto)
 | 
				
			||||||
endif()
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if(WIN32)
 | 
					if(WIN32 OR MINGW)
 | 
				
			||||||
    add_definitions(-DWIN32_LEAN_AND_MEAN -D_CRT_SECURE_NO_WARNINGS -D_WIN32_WINNT=0x0600)
 | 
					    add_definitions(-DWIN32_LEAN_AND_MEAN -D_CRT_SECURE_NO_WARNINGS -D_WIN32_WINNT=0x0600)
 | 
				
			||||||
    set(LIBS ${LIBS} secur32 crypt32 winmm iphlpapi ws2_32)
 | 
					    set(LIBS ${LIBS} secur32 crypt32 winmm iphlpapi ws2_32)
 | 
				
			||||||
    if(ENABLE_WINDUMP)
 | 
					    if(ENABLE_WINDUMP)
 | 
				
			||||||
| 
						 | 
					@ -168,9 +176,10 @@ endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if(ANDROID)
 | 
					if(ANDROID)
 | 
				
			||||||
    set(LIBS ${LIBS} log)
 | 
					    set(LIBS ${LIBS} log)
 | 
				
			||||||
elseif(UNIX)
 | 
					elseif(UNIX AND NOT MINGW)
 | 
				
			||||||
    set(LIBS ${LIBS} pthread m dl)
 | 
					    set(LIBS ${LIBS} pthread m dl)
 | 
				
			||||||
    if(CMAKE_COMPILER_IS_GNUCC)
 | 
					    find_library(RT_LIBRARY rt)
 | 
				
			||||||
 | 
					    if(RT_LIBRARY)
 | 
				
			||||||
        set(LIBS ${LIBS} rt)
 | 
					        set(LIBS ${LIBS} rt)
 | 
				
			||||||
    endif()
 | 
					    endif()
 | 
				
			||||||
endif()
 | 
					endif()
 | 
				
			||||||
| 
						 | 
					@ -182,7 +191,7 @@ endif()
 | 
				
			||||||
# see Makefile
 | 
					# see Makefile
 | 
				
			||||||
set(ALL_SRCDIRS . base ssl event event/kcp util cpputil evpp protocol http http/client http/server mqtt)
 | 
					set(ALL_SRCDIRS . base ssl event event/kcp util cpputil evpp protocol http http/client http/server mqtt)
 | 
				
			||||||
set(CORE_SRCDIRS . base ssl event)
 | 
					set(CORE_SRCDIRS . base ssl event)
 | 
				
			||||||
if(WIN32)
 | 
					if(WIN32 OR MINGW)
 | 
				
			||||||
    if(WITH_WEPOLL)
 | 
					    if(WITH_WEPOLL)
 | 
				
			||||||
        set(CORE_SRCDIRS ${CORE_SRCDIRS} event/wepoll)
 | 
					        set(CORE_SRCDIRS ${CORE_SRCDIRS} event/wepoll)
 | 
				
			||||||
    endif()
 | 
					    endif()
 | 
				
			||||||
| 
						 | 
					@ -280,3 +289,24 @@ endif()
 | 
				
			||||||
if(BUILD_UNITTEST)
 | 
					if(BUILD_UNITTEST)
 | 
				
			||||||
    add_subdirectory(unittest)
 | 
					    add_subdirectory(unittest)
 | 
				
			||||||
endif()
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# CPack settings
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_NAME "libhv")
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_VERSION "${PROJECT_VERSION}")
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_RELEASE 1)
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_DESCRIPTION_SUMMARY "A high-performance C/C++ network library")
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_VENDOR "libhv")
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_CONTACT "ithewei <ithewei@163.com>")
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_FILE_NAME "${CPACK_PACKAGE_NAME}-${CPACK_PACKAGE_VERSION}-${CMAKE_HOST_SYSTEM_PROCESSOR}")
 | 
				
			||||||
 | 
					set(CPACK_RESOURCE_FILE_LICENSE "${CMAKE_CURRENT_SOURCE_DIR}/LICENSE")
 | 
				
			||||||
 | 
					set(CPACK_RESOURCE_FILE_README "${CMAKE_CURRENT_SOURCE_DIR}/README.md")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Specify the package generators
 | 
				
			||||||
 | 
					set(CPACK_GENERATOR "TGZ;DEB;RPM")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Enable CPack debug output
 | 
				
			||||||
 | 
					set(CPACK_PACKAGE_DEBUG True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# https://cmake.org/cmake/help/latest/variable/CPACK_ERROR_ON_ABSOLUTE_INSTALL_DESTINATION.html
 | 
				
			||||||
 | 
					set(CPACK_ERROR_ON_ABSOLUTE_INSTALL_DESTINATION "ON")
 | 
				
			||||||
 | 
					include(CPack)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -52,7 +52,7 @@ default: all
 | 
				
			||||||
all: libhv examples
 | 
					all: libhv examples
 | 
				
			||||||
	@echo "make all done, please enjoy libhv."
 | 
						@echo "make all done, please enjoy libhv."
 | 
				
			||||||
 | 
					
 | 
				
			||||||
examples: hmain_test htimer_test hloop_test \
 | 
					examples: hmain_test htimer_test hloop_test pipe_test \
 | 
				
			||||||
	nc nmap tinyhttpd tinyproxyd httpd curl wget wrk consul \
 | 
						nc nmap tinyhttpd tinyproxyd httpd curl wget wrk consul \
 | 
				
			||||||
	tcp_client_test \
 | 
						tcp_client_test \
 | 
				
			||||||
	tcp_echo_server \
 | 
						tcp_echo_server \
 | 
				
			||||||
| 
						 | 
					@ -111,6 +111,9 @@ htimer_test: prepare
 | 
				
			||||||
hloop_test: prepare
 | 
					hloop_test: prepare
 | 
				
			||||||
	$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS)" SRCS="examples/hloop_test.c"
 | 
						$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS)" SRCS="examples/hloop_test.c"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pipe_test: prepare
 | 
				
			||||||
 | 
						$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS)" SRCS="examples/pipe_test.c"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
tcp_client_test: prepare
 | 
					tcp_client_test: prepare
 | 
				
			||||||
	$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS)" SRCS="examples/tcp_client_test.c"
 | 
						$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS)" SRCS="examples/tcp_client_test.c"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -191,6 +194,14 @@ mqtt_pub: prepare
 | 
				
			||||||
mqtt_client_test: prepare
 | 
					mqtt_client_test: prepare
 | 
				
			||||||
	$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS) mqtt" SRCS="examples/mqtt/mqtt_client_test.cpp"
 | 
						$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS) mqtt" SRCS="examples/mqtt/mqtt_client_test.cpp"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					kcptun: kcptun_client kcptun_server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					kcptun_client: prepare
 | 
				
			||||||
 | 
						$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS) examples/kcptun/smux examples/kcptun/client"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					kcptun_server: prepare
 | 
				
			||||||
 | 
						$(MAKEF) TARGET=$@ SRCDIRS="$(CORE_SRCDIRS) examples/kcptun/smux examples/kcptun/server"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
jsonrpc: jsonrpc_client jsonrpc_server
 | 
					jsonrpc: jsonrpc_client jsonrpc_server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
jsonrpc_client: prepare
 | 
					jsonrpc_client: prepare
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -52,7 +52,7 @@
 | 
				
			||||||
## ✨ 特性
 | 
					## ✨ 特性
 | 
				
			||||||
 | 
					
 | 
				
			||||||
- 跨平台(Linux, Windows, macOS, Android, iOS, BSD, Solaris)
 | 
					- 跨平台(Linux, Windows, macOS, Android, iOS, BSD, Solaris)
 | 
				
			||||||
- 高性能事件循环(网络IO事件、定时器事件、空闲事件、自定义事件)
 | 
					- 高性能事件循环(网络IO事件、定时器事件、空闲事件、自定义事件、信号)
 | 
				
			||||||
- TCP/UDP服务端/客户端/代理
 | 
					- TCP/UDP服务端/客户端/代理
 | 
				
			||||||
- TCP支持心跳、重连、转发、多线程安全write和close等特性
 | 
					- TCP支持心跳、重连、转发、多线程安全write和close等特性
 | 
				
			||||||
- 内置常见的拆包模式(固定包长、分界符、头部长度字段)
 | 
					- 内置常见的拆包模式(固定包长、分界符、头部长度字段)
 | 
				
			||||||
| 
						 | 
					@ -402,6 +402,7 @@ int main(int argc, char** argv) {
 | 
				
			||||||
### c版本
 | 
					### c版本
 | 
				
			||||||
- 事件循环:     [examples/hloop_test.c](examples/hloop_test.c)
 | 
					- 事件循环:     [examples/hloop_test.c](examples/hloop_test.c)
 | 
				
			||||||
- 定时器:       [examples/htimer_test.c](examples/htimer_test.c)
 | 
					- 定时器:       [examples/htimer_test.c](examples/htimer_test.c)
 | 
				
			||||||
 | 
					- pipe示例:     [examples/pipe_test.c](examples/pipe_test.c)
 | 
				
			||||||
- TCP回显服务:  [examples/tcp_echo_server.c](examples/tcp_echo_server.c)
 | 
					- TCP回显服务:  [examples/tcp_echo_server.c](examples/tcp_echo_server.c)
 | 
				
			||||||
- TCP聊天服务:  [examples/tcp_chat_server.c](examples/tcp_chat_server.c)
 | 
					- TCP聊天服务:  [examples/tcp_chat_server.c](examples/tcp_chat_server.c)
 | 
				
			||||||
- TCP代理服务:  [examples/tcp_proxy_server.c](examples/tcp_proxy_server.c)
 | 
					- TCP代理服务:  [examples/tcp_proxy_server.c](examples/tcp_proxy_server.c)
 | 
				
			||||||
| 
						 | 
					@ -440,6 +441,7 @@ int main(int argc, char** argv) {
 | 
				
			||||||
- URL请求工具:  [examples/curl](examples/curl.cpp)
 | 
					- URL请求工具:  [examples/curl](examples/curl.cpp)
 | 
				
			||||||
- 文件下载工具: [examples/wget](examples/wget.cpp)
 | 
					- 文件下载工具: [examples/wget](examples/wget.cpp)
 | 
				
			||||||
- 服务注册与发现: [examples/consul](examples/consul)
 | 
					- 服务注册与发现: [examples/consul](examples/consul)
 | 
				
			||||||
 | 
					- kcptun隧道: [examples/kcptun](examples/kcptun)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## 🥇 性能测试
 | 
					## 🥇 性能测试
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -28,7 +28,7 @@ but simpler api and richer protocols.
 | 
				
			||||||
## ✨ Features
 | 
					## ✨ Features
 | 
				
			||||||
 | 
					
 | 
				
			||||||
- Cross-platform (Linux, Windows, macOS, Android, iOS, BSD, Solaris)
 | 
					- Cross-platform (Linux, Windows, macOS, Android, iOS, BSD, Solaris)
 | 
				
			||||||
- High-performance EventLoop (IO, timer, idle, custom)
 | 
					- High-performance EventLoop (IO, timer, idle, custom, signal)
 | 
				
			||||||
- TCP/UDP client/server/proxy
 | 
					- TCP/UDP client/server/proxy
 | 
				
			||||||
- TCP supports heartbeat, reconnect, upstream, MultiThread-safe write and close, etc.
 | 
					- TCP supports heartbeat, reconnect, upstream, MultiThread-safe write and close, etc.
 | 
				
			||||||
- Built-in common unpacking modes (FixedLength, Delimiter, LengthField)
 | 
					- Built-in common unpacking modes (FixedLength, Delimiter, LengthField)
 | 
				
			||||||
| 
						 | 
					@ -342,6 +342,7 @@ int main(int argc, char** argv) {
 | 
				
			||||||
### c version
 | 
					### c version
 | 
				
			||||||
- [examples/hloop_test.c](examples/hloop_test.c)
 | 
					- [examples/hloop_test.c](examples/hloop_test.c)
 | 
				
			||||||
- [examples/htimer_test.c](examples/htimer_test.c)
 | 
					- [examples/htimer_test.c](examples/htimer_test.c)
 | 
				
			||||||
 | 
					- [examples/pipe_test.c](examples/pipe_test.c)
 | 
				
			||||||
- [examples/tcp_echo_server.c](examples/tcp_echo_server.c)
 | 
					- [examples/tcp_echo_server.c](examples/tcp_echo_server.c)
 | 
				
			||||||
- [examples/tcp_chat_server.c](examples/tcp_chat_server.c)
 | 
					- [examples/tcp_chat_server.c](examples/tcp_chat_server.c)
 | 
				
			||||||
- [examples/tcp_proxy_server.c](examples/tcp_proxy_server.c)
 | 
					- [examples/tcp_proxy_server.c](examples/tcp_proxy_server.c)
 | 
				
			||||||
| 
						 | 
					@ -380,6 +381,7 @@ int main(int argc, char** argv) {
 | 
				
			||||||
- [examples/curl](examples/curl.cpp)
 | 
					- [examples/curl](examples/curl.cpp)
 | 
				
			||||||
- [examples/wget](examples/wget.cpp)
 | 
					- [examples/wget](examples/wget.cpp)
 | 
				
			||||||
- [examples/consul](examples/consul)
 | 
					- [examples/consul](examples/consul)
 | 
				
			||||||
 | 
					- [examples/kcptun](examples/kcptun)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## 🥇 Benchmark
 | 
					## 🥇 Benchmark
 | 
				
			||||||
### `pingpong echo-servers`
 | 
					### `pingpong echo-servers`
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -195,6 +195,17 @@ char* hv_strnchr(const char* s, char c, size_t n) {
 | 
				
			||||||
    return NULL;
 | 
					    return NULL;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					char* hv_strnrchr(const char* s, char c, size_t n) {
 | 
				
			||||||
 | 
					    assert(s != NULL);
 | 
				
			||||||
 | 
					    const char* p = s;
 | 
				
			||||||
 | 
					    const char* last = NULL;
 | 
				
			||||||
 | 
					    while (*p != '\0' && n-- > 0) {
 | 
				
			||||||
 | 
					        if (*p == c) last = p;
 | 
				
			||||||
 | 
					        ++p;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return (char*)last;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
char* hv_strrchr_dir(const char* filepath) {
 | 
					char* hv_strrchr_dir(const char* filepath) {
 | 
				
			||||||
    char* p = (char*)filepath;
 | 
					    char* p = (char*)filepath;
 | 
				
			||||||
    while (*p) ++p;
 | 
					    while (*p) ++p;
 | 
				
			||||||
| 
						 | 
					@ -473,8 +484,19 @@ int hv_parse_url(hurl_t* stURL, const char* strURL) {
 | 
				
			||||||
        // @
 | 
					        // @
 | 
				
			||||||
        host = pos + 1;
 | 
					        host = pos + 1;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    // port
 | 
					    // host:port or ipv4:port or [ipv6]:port
 | 
				
			||||||
    const char* port = hv_strnchr(host, ':', ep - host);
 | 
					    const char* hostend = host;
 | 
				
			||||||
 | 
					    if (*host == '[') {
 | 
				
			||||||
 | 
					        pos = hv_strnchr(host, ']', ep - host);
 | 
				
			||||||
 | 
					        if (pos) {
 | 
				
			||||||
 | 
					            // ipv6
 | 
				
			||||||
 | 
					            host++;
 | 
				
			||||||
 | 
					            hostend = pos;
 | 
				
			||||||
 | 
					            stURL->fields[HV_URL_HOST].off = host - begin;
 | 
				
			||||||
 | 
					            stURL->fields[HV_URL_HOST].len = hostend - host;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    const char* port = hv_strnchr(hostend, ':', ep - hostend);
 | 
				
			||||||
    if (port) {
 | 
					    if (port) {
 | 
				
			||||||
        stURL->fields[HV_URL_PORT].off = port + 1 - begin;
 | 
					        stURL->fields[HV_URL_PORT].off = port + 1 - begin;
 | 
				
			||||||
        stURL->fields[HV_URL_PORT].len = ep - port - 1;
 | 
					        stURL->fields[HV_URL_PORT].len = ep - port - 1;
 | 
				
			||||||
| 
						 | 
					@ -492,9 +514,10 @@ int hv_parse_url(hurl_t* stURL, const char* strURL) {
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    // host
 | 
					    if (stURL->fields[HV_URL_HOST].len == 0) {
 | 
				
			||||||
        stURL->fields[HV_URL_HOST].off = host - begin;
 | 
					        stURL->fields[HV_URL_HOST].off = host - begin;
 | 
				
			||||||
        stURL->fields[HV_URL_HOST].len = port - host;
 | 
					        stURL->fields[HV_URL_HOST].len = port - host;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    if (ep == end) return 0;
 | 
					    if (ep == end) return 0;
 | 
				
			||||||
    // /path
 | 
					    // /path
 | 
				
			||||||
    sp = ep;
 | 
					    sp = ep;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -82,6 +82,7 @@ HV_EXPORT char* hv_strncat(char* dest, const char* src, size_t n);
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
HV_EXPORT char* hv_strnchr(const char* s, char c, size_t n);
 | 
					HV_EXPORT char* hv_strnchr(const char* s, char c, size_t n);
 | 
				
			||||||
 | 
					HV_EXPORT char* hv_strnrchr(const char* s, char c, size_t n);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define hv_strrchr_dot(str) strrchr(str, '.')
 | 
					#define hv_strrchr_dot(str) strrchr(str, '.')
 | 
				
			||||||
HV_EXPORT char* hv_strrchr_dir(const char* filepath);
 | 
					HV_EXPORT char* hv_strrchr_dir(const char* filepath);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,4 +1,4 @@
 | 
				
			||||||
#ifndef HV_LOG_H_
 | 
					#ifndef HV_LOG_H_
 | 
				
			||||||
#define HV_LOG_H_
 | 
					#define HV_LOG_H_
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/*
 | 
					/*
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -316,7 +316,7 @@ int parse_opt_long(int argc, char** argv, const option_t* long_options, int size
 | 
				
			||||||
        if (pOption->arg_type == NO_ARGUMENT) {
 | 
					        if (pOption->arg_type == NO_ARGUMENT) {
 | 
				
			||||||
            // -h
 | 
					            // -h
 | 
				
			||||||
            value = OPTION_ENABLE;
 | 
					            value = OPTION_ENABLE;
 | 
				
			||||||
        } else if (pOption->arg_type == REQUIRED_ARGUMENT) {
 | 
					        } else {
 | 
				
			||||||
            if (delim) {
 | 
					            if (delim) {
 | 
				
			||||||
                // --port=80
 | 
					                // --port=80
 | 
				
			||||||
                value = delim+1;
 | 
					                value = delim+1;
 | 
				
			||||||
| 
						 | 
					@ -327,9 +327,12 @@ int parse_opt_long(int argc, char** argv, const option_t* long_options, int size
 | 
				
			||||||
                } else if (argv[i+1] != NULL) {
 | 
					                } else if (argv[i+1] != NULL) {
 | 
				
			||||||
                    // --port 80
 | 
					                    // --port 80
 | 
				
			||||||
                    value = argv[++i];
 | 
					                    value = argv[++i];
 | 
				
			||||||
                } else {
 | 
					                } else if (pOption->arg_type == REQUIRED_ARGUMENT) {
 | 
				
			||||||
                    printf("Option '%s' requires parament\n", opt);
 | 
					                    printf("Option '%s' requires parament\n", opt);
 | 
				
			||||||
                    return -20;
 | 
					                    return -20;
 | 
				
			||||||
 | 
					                } else {
 | 
				
			||||||
 | 
					                    // arg_type == OPTIONAL_ARGUMENT
 | 
				
			||||||
 | 
					                    value = OPTION_ENABLE;
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
| 
						 | 
					@ -343,6 +346,69 @@ int parse_opt_long(int argc, char** argv, const option_t* long_options, int size
 | 
				
			||||||
    return 0;
 | 
					    return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int dump_opt_long(const option_t* long_options, int opt_size, char* out_str, int out_size) {
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * Usage: program_name [short_options]
 | 
				
			||||||
 | 
					     * Options:
 | 
				
			||||||
 | 
					     *   -%c|--%s          description
 | 
				
			||||||
 | 
					     *   -%c|--%s <value>  description
 | 
				
			||||||
 | 
					     *   -%c|--%s [value]  description
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    int align = 0, max_align = 0;
 | 
				
			||||||
 | 
					    char short_options[256] = {0};
 | 
				
			||||||
 | 
					    char* p = short_options;
 | 
				
			||||||
 | 
					    for (int i = 0; i < opt_size; ++i) {
 | 
				
			||||||
 | 
					        if (long_options[i].short_opt > 0) {
 | 
				
			||||||
 | 
					            *p++ = long_options[i].short_opt;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (long_options[i].arg_type == NO_ARGUMENT) {
 | 
				
			||||||
 | 
					            // "  -%c|--%s  "
 | 
				
			||||||
 | 
					            align = 9 + strlen(long_options[i].long_opt);
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            // "  -%c|--%s <value>  "
 | 
				
			||||||
 | 
					            align = 17 + strlen(long_options[i].long_opt);
 | 
				
			||||||
 | 
					            if (long_options[i].short_opt > 0) {
 | 
				
			||||||
 | 
					                *p++ = ':';
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (align > max_align) max_align = align;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    int offset = 0;
 | 
				
			||||||
 | 
					    if (*g_main_ctx.program_name) {
 | 
				
			||||||
 | 
					        offset = snprintf(out_str, out_size, "Usage: %s [%s]\n", g_main_ctx.program_name, short_options);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    offset += snprintf(out_str + offset, out_size - offset, "Options:\n");
 | 
				
			||||||
 | 
					    char short_opt_chars[4] = {0};
 | 
				
			||||||
 | 
					    for (int i = 0; i < opt_size; ++i) {
 | 
				
			||||||
 | 
					        if (long_options[i].short_opt > 0) {
 | 
				
			||||||
 | 
					            // -%c|
 | 
				
			||||||
 | 
					            short_opt_chars[0] = '-';
 | 
				
			||||||
 | 
					            short_opt_chars[1] = long_options[i].short_opt;
 | 
				
			||||||
 | 
					            short_opt_chars[2] = '|';
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            short_opt_chars[0] = ' ';
 | 
				
			||||||
 | 
					            short_opt_chars[1] = ' ';
 | 
				
			||||||
 | 
					            short_opt_chars[2] = ' ';
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (long_options[i].arg_type == NO_ARGUMENT) {
 | 
				
			||||||
 | 
					            // "  -%c|--%s  "
 | 
				
			||||||
 | 
					            align = 9 + strlen(long_options[i].long_opt);
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            // "  -%c|--%s <value>  "
 | 
				
			||||||
 | 
					            align = 17 + strlen(long_options[i].long_opt);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        offset += snprintf(out_str + offset, out_size - offset, "  %s--%s%s  %*s%s\n",
 | 
				
			||||||
 | 
					                short_opt_chars,
 | 
				
			||||||
 | 
					                long_options[i].long_opt,
 | 
				
			||||||
 | 
					                long_options[i].arg_type == REQUIRED_ARGUMENT ? " <value>" :
 | 
				
			||||||
 | 
					                long_options[i].arg_type == OPTIONAL_ARGUMENT ? " [value]" : "",
 | 
				
			||||||
 | 
					                max_align - align, "",
 | 
				
			||||||
 | 
					                long_options[i].description ? long_options[i].description : "");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return offset;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if defined(OS_UNIX) && !HAVE_SETPROCTITLE
 | 
					#if defined(OS_UNIX) && !HAVE_SETPROCTITLE
 | 
				
			||||||
/*
 | 
					/*
 | 
				
			||||||
 * memory layout
 | 
					 * memory layout
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -65,6 +65,7 @@ typedef struct option_s {
 | 
				
			||||||
    char        short_opt;
 | 
					    char        short_opt;
 | 
				
			||||||
    const char* long_opt;
 | 
					    const char* long_opt;
 | 
				
			||||||
    int         arg_type;
 | 
					    int         arg_type;
 | 
				
			||||||
 | 
					    const char* description;
 | 
				
			||||||
} option_t;
 | 
					} option_t;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
HV_EXPORT int  main_ctx_init(int argc, char** argv);
 | 
					HV_EXPORT int  main_ctx_init(int argc, char** argv);
 | 
				
			||||||
| 
						 | 
					@ -76,7 +77,8 @@ HV_EXPORT void main_ctx_free(void);
 | 
				
			||||||
// watch -n10 ls
 | 
					// watch -n10 ls
 | 
				
			||||||
HV_EXPORT int parse_opt(int argc, char** argv, const char* opt);
 | 
					HV_EXPORT int parse_opt(int argc, char** argv, const char* opt);
 | 
				
			||||||
// gcc -g -Wall -O3 -std=cpp main.c
 | 
					// gcc -g -Wall -O3 -std=cpp main.c
 | 
				
			||||||
HV_EXPORT int parse_opt_long(int argc, char** argv, const option_t* long_options, int size);
 | 
					HV_EXPORT int parse_opt_long(int argc, char** argv, const option_t* long_options, int opt_size);
 | 
				
			||||||
 | 
					HV_EXPORT int dump_opt_long(const option_t* long_options, int opt_size, char* out_str, int out_size);
 | 
				
			||||||
HV_EXPORT const char* get_arg(const char* key);
 | 
					HV_EXPORT const char* get_arg(const char* key);
 | 
				
			||||||
HV_EXPORT const char* get_env(const char* key);
 | 
					HV_EXPORT const char* get_env(const char* key);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -41,6 +41,7 @@
 | 
				
			||||||
    #undef  OS_UNIX
 | 
					    #undef  OS_UNIX
 | 
				
			||||||
    #define OS_WIN
 | 
					    #define OS_WIN
 | 
				
			||||||
#else
 | 
					#else
 | 
				
			||||||
 | 
					    #undef  OS_WIN
 | 
				
			||||||
    #define OS_UNIX
 | 
					    #define OS_UNIX
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -142,6 +143,12 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// headers
 | 
					// headers
 | 
				
			||||||
#ifdef OS_WIN
 | 
					#ifdef OS_WIN
 | 
				
			||||||
 | 
					    #ifndef _WIN32_WINNT
 | 
				
			||||||
 | 
					    #define _WIN32_WINNT 0x0600
 | 
				
			||||||
 | 
					    #elif _WIN32_WINNT < 0x0600
 | 
				
			||||||
 | 
					    #undef _WIN32_WINNT
 | 
				
			||||||
 | 
					    #define _WIN32_WINNT 0x0600
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
    #ifndef WIN32_LEAN_AND_MEAN
 | 
					    #ifndef WIN32_LEAN_AND_MEAN
 | 
				
			||||||
    #define WIN32_LEAN_AND_MEAN
 | 
					    #define WIN32_LEAN_AND_MEAN
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -82,9 +82,11 @@ HV_INLINE int nonblocking(int s) {
 | 
				
			||||||
    return fcntl(s, F_SETFL, fcntl(s, F_GETFL) |  O_NONBLOCK);
 | 
					    return fcntl(s, F_SETFL, fcntl(s, F_GETFL) |  O_NONBLOCK);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef closesocket
 | 
				
			||||||
HV_INLINE int closesocket(int sockfd) {
 | 
					HV_INLINE int closesocket(int sockfd) {
 | 
				
			||||||
    return close(sockfd);
 | 
					    return close(sockfd);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -8,7 +8,7 @@ BEGIN_EXTERN_C
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define HV_VERSION_MAJOR    1
 | 
					#define HV_VERSION_MAJOR    1
 | 
				
			||||||
#define HV_VERSION_MINOR    3
 | 
					#define HV_VERSION_MINOR    3
 | 
				
			||||||
#define HV_VERSION_PATCH    2
 | 
					#define HV_VERSION_PATCH    3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define HV_VERSION_STRING   STRINGIFY(HV_VERSION_MAJOR) "." \
 | 
					#define HV_VERSION_STRING   STRINGIFY(HV_VERSION_MAJOR) "." \
 | 
				
			||||||
                            STRINGIFY(HV_VERSION_MINOR) "." \
 | 
					                            STRINGIFY(HV_VERSION_MINOR) "." \
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -71,7 +71,7 @@ std::string HPath::filename(const std::string& filepath) {
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
        pos1++;
 | 
					        pos1++;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    std::string file = filepath.substr(pos1, -1);
 | 
					    std::string file = filepath.substr(pos1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    std::string::size_type pos2 = file.find_last_of(".");
 | 
					    std::string::size_type pos2 = file.find_last_of(".");
 | 
				
			||||||
    if (pos2 == std::string::npos) {
 | 
					    if (pos2 == std::string::npos) {
 | 
				
			||||||
| 
						 | 
					@ -87,13 +87,13 @@ std::string HPath::suffixname(const std::string& filepath) {
 | 
				
			||||||
    } else {
 | 
					    } else {
 | 
				
			||||||
        pos1++;
 | 
					        pos1++;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    std::string file = filepath.substr(pos1, -1);
 | 
					    std::string file = filepath.substr(pos1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    std::string::size_type pos2 = file.find_last_of(".");
 | 
					    std::string::size_type pos2 = file.find_last_of(".");
 | 
				
			||||||
    if (pos2 == std::string::npos) {
 | 
					    if (pos2 == std::string::npos) {
 | 
				
			||||||
        return "";
 | 
					        return "";
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    return file.substr(pos2+1, -1);
 | 
					    return file.substr(pos2+1);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
std::string HPath::join(const std::string& dir, const std::string& filename) {
 | 
					std::string HPath::join(const std::string& dir, const std::string& filename) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -163,7 +163,7 @@ const std::string& HUrl::dump() {
 | 
				
			||||||
        if (port != 80 && port != 443) {
 | 
					        if (port != 80 && port != 443) {
 | 
				
			||||||
            char buf[16] = {0};
 | 
					            char buf[16] = {0};
 | 
				
			||||||
            snprintf(buf, sizeof(buf), ":%d", port);
 | 
					            snprintf(buf, sizeof(buf), ":%d", port);
 | 
				
			||||||
            url += port;
 | 
					            url += buf;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    // /path
 | 
					    // /path
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -287,24 +287,24 @@ int IniParser::SaveAs(const char* filepath) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
std::list<std::string> IniParser::GetSections() {
 | 
					std::list<std::string> IniParser::GetSections() {
 | 
				
			||||||
    std::list<std::string> ret;
 | 
					    std::list<std::string> ret;
 | 
				
			||||||
    if (root_ == NULL) return std::move(ret);
 | 
					    if (root_ == NULL) return ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for (auto pNode : root_->children) {
 | 
					    for (auto pNode : root_->children) {
 | 
				
			||||||
        if (pNode->type == IniNode::INI_NODE_TYPE_SECTION) {
 | 
					        if (pNode->type == IniNode::INI_NODE_TYPE_SECTION) {
 | 
				
			||||||
            ret.push_back(pNode->label);
 | 
					            ret.push_back(pNode->label);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    return std::move(ret);
 | 
					    return ret;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
std::list<std::string> IniParser::GetKeys(const std::string& section) {
 | 
					std::list<std::string> IniParser::GetKeys(const std::string& section) {
 | 
				
			||||||
    std::list<std::string> ret;
 | 
					    std::list<std::string> ret;
 | 
				
			||||||
    if (root_ == NULL) return std::move(ret);
 | 
					    if (root_ == NULL) return ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    IniNode* pSection = root_;
 | 
					    IniNode* pSection = root_;
 | 
				
			||||||
    if (section.length() != 0) {
 | 
					    if (section.length() != 0) {
 | 
				
			||||||
        pSection = root_->Get(section, IniNode::INI_NODE_TYPE_SECTION);
 | 
					        pSection = root_->Get(section, IniNode::INI_NODE_TYPE_SECTION);
 | 
				
			||||||
        if (pSection == NULL) return std::move(ret);
 | 
					        if (pSection == NULL) return ret;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for (auto pNode : pSection->children) {
 | 
					    for (auto pNode : pSection->children) {
 | 
				
			||||||
| 
						 | 
					@ -312,7 +312,7 @@ std::list<std::string> IniParser::GetKeys(const std::string& section) {
 | 
				
			||||||
            ret.push_back(pNode->label);
 | 
					            ret.push_back(pNode->label);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    return std::move(ret);
 | 
					    return ret;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
std::string IniParser::GetValue(const std::string& key, const std::string& section) {
 | 
					std::string IniParser::GetValue(const std::string& key, const std::string& section) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -269,6 +269,7 @@
 | 
				
			||||||
- main_ctx_init
 | 
					- main_ctx_init
 | 
				
			||||||
- parse_opt
 | 
					- parse_opt
 | 
				
			||||||
- parse_opt_long
 | 
					- parse_opt_long
 | 
				
			||||||
 | 
					- dump_opt_long
 | 
				
			||||||
- get_arg
 | 
					- get_arg
 | 
				
			||||||
- get_env
 | 
					- get_env
 | 
				
			||||||
- setproctitle
 | 
					- setproctitle
 | 
				
			||||||
| 
						 | 
					@ -470,6 +471,7 @@
 | 
				
			||||||
- hio_setup_ssl_upstream
 | 
					- hio_setup_ssl_upstream
 | 
				
			||||||
- hio_setup_udp_upstream
 | 
					- hio_setup_udp_upstream
 | 
				
			||||||
- hio_create_socket
 | 
					- hio_create_socket
 | 
				
			||||||
 | 
					- hio_create_pipe
 | 
				
			||||||
- hio_context
 | 
					- hio_context
 | 
				
			||||||
- hio_set_context
 | 
					- hio_set_context
 | 
				
			||||||
- htimer_add
 | 
					- htimer_add
 | 
				
			||||||
| 
						 | 
					@ -478,6 +480,8 @@
 | 
				
			||||||
- htimer_reset
 | 
					- htimer_reset
 | 
				
			||||||
- hidle_add
 | 
					- hidle_add
 | 
				
			||||||
- hidle_del
 | 
					- hidle_del
 | 
				
			||||||
 | 
					- hsignal_add
 | 
				
			||||||
 | 
					- hsignal_del
 | 
				
			||||||
 | 
					
 | 
				
			||||||
### nlog.h
 | 
					### nlog.h
 | 
				
			||||||
- network_logger
 | 
					- network_logger
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -43,7 +43,7 @@ struct hevent_s {
 | 
				
			||||||
// 获取事件用户数据
 | 
					// 获取事件用户数据
 | 
				
			||||||
#define hevent_userdata(ev)     (((hevent_t*)(ev))->userdata)
 | 
					#define hevent_userdata(ev)     (((hevent_t*)(ev))->userdata)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// hidle_t、htimer_t、hio_t皆是继承自hevent_t,继承上面的数据成员和函数方法
 | 
					// hio_t、htimer_t、hsignal_t、hidle_t皆是继承自hevent_t,继承上面的数据成员和函数方法
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// 新建事件循环
 | 
					// 新建事件循环
 | 
				
			||||||
hloop_t* hloop_new(int flags DEFAULT(HLOOP_FLAG_AUTO_FREE));
 | 
					hloop_t* hloop_new(int flags DEFAULT(HLOOP_FLAG_AUTO_FREE));
 | 
				
			||||||
| 
						 | 
					@ -107,6 +107,12 @@ void* hloop_userdata(hloop_t* loop);
 | 
				
			||||||
// 投递事件
 | 
					// 投递事件
 | 
				
			||||||
void hloop_post_event(hloop_t* loop, hevent_t* ev);
 | 
					void hloop_post_event(hloop_t* loop, hevent_t* ev);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// 添加信号处理
 | 
				
			||||||
 | 
					hsignal_t* hsignal_add(hloop_t* loop, hsignal_cb cb, int signo);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// 删除信号处理
 | 
				
			||||||
 | 
					void       hsignal_del(hsignal_t* sig);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// 添加空闲事件
 | 
					// 添加空闲事件
 | 
				
			||||||
hidle_t* hidle_add(hloop_t* loop, hidle_cb cb, uint32_t repeat DEFAULT(INFINITE));
 | 
					hidle_t* hidle_add(hloop_t* loop, hidle_cb cb, uint32_t repeat DEFAULT(INFINITE));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -416,6 +422,10 @@ hio_t* hloop_create_udp_server (hloop_t* loop, const char* host, int port);
 | 
				
			||||||
// 创建UDP客户端,示例代码见 examples/nc.c
 | 
					// 创建UDP客户端,示例代码见 examples/nc.c
 | 
				
			||||||
hio_t* hloop_create_udp_client (hloop_t* loop, const char* host, int port);
 | 
					hio_t* hloop_create_udp_client (hloop_t* loop, const char* host, int port);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//-----------------pipe---------------------------------------------
 | 
				
			||||||
 | 
					// 创建pipe,示例代码见 examples/pipe_test.c
 | 
				
			||||||
 | 
					int hio_create_pipe(hloop_t* loop, hio_t* pipeio[2]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//-----------------转发---------------------------------------------
 | 
					//-----------------转发---------------------------------------------
 | 
				
			||||||
// hio_read(io)
 | 
					// hio_read(io)
 | 
				
			||||||
// hio_read(io->upstream_io)
 | 
					// hio_read(io->upstream_io)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -400,7 +400,7 @@ void hio_read_cb(hio_t* io, void* buf, int len) {
 | 
				
			||||||
        hio_read_stop(io);
 | 
					        hio_read_stop(io);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (io->read_cb) {
 | 
					    if (io->read_cb && !io->closed) {
 | 
				
			||||||
        // printd("read_cb------\n");
 | 
					        // printd("read_cb------\n");
 | 
				
			||||||
        io->read_cb(io, buf, len);
 | 
					        io->read_cb(io, buf, len);
 | 
				
			||||||
        // printd("read_cb======\n");
 | 
					        // printd("read_cb======\n");
 | 
				
			||||||
| 
						 | 
					@ -418,7 +418,7 @@ void hio_read_cb(hio_t* io, void* buf, int len) {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void hio_write_cb(hio_t* io, const void* buf, int len) {
 | 
					void hio_write_cb(hio_t* io, const void* buf, int len) {
 | 
				
			||||||
    if (io->write_cb) {
 | 
					    if (io->write_cb  && !io->closed) {
 | 
				
			||||||
        // printd("write_cb------\n");
 | 
					        // printd("write_cb------\n");
 | 
				
			||||||
        io->write_cb(io, buf, len);
 | 
					        io->write_cb(io, buf, len);
 | 
				
			||||||
        // printd("write_cb======\n");
 | 
					        // printd("write_cb======\n");
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -25,6 +25,7 @@
 | 
				
			||||||
#define HIO_READ_UNTIL_DELIM    0x4
 | 
					#define HIO_READ_UNTIL_DELIM    0x4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ARRAY_DECL(hio_t*, io_array);
 | 
					ARRAY_DECL(hio_t*, io_array);
 | 
				
			||||||
 | 
					ARRAY_DECL(hsignal_t*, signal_array);
 | 
				
			||||||
QUEUE_DECL(hevent_t, event_queue);
 | 
					QUEUE_DECL(hevent_t, event_queue);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct hloop_s {
 | 
					struct hloop_s {
 | 
				
			||||||
| 
						 | 
					@ -45,6 +46,9 @@ struct hloop_s {
 | 
				
			||||||
    uint32_t                    npendings;
 | 
					    uint32_t                    npendings;
 | 
				
			||||||
    // pendings: with priority as array.index
 | 
					    // pendings: with priority as array.index
 | 
				
			||||||
    hevent_t*                   pendings[HEVENT_PRIORITY_SIZE];
 | 
					    hevent_t*                   pendings[HEVENT_PRIORITY_SIZE];
 | 
				
			||||||
 | 
					    // signals
 | 
				
			||||||
 | 
					    struct signal_array         signals;
 | 
				
			||||||
 | 
					    uint32_t                    nsignals;
 | 
				
			||||||
    // idles
 | 
					    // idles
 | 
				
			||||||
    struct list_head            idles;
 | 
					    struct list_head            idles;
 | 
				
			||||||
    uint32_t                    nidles;
 | 
					    uint32_t                    nidles;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -394,6 +394,14 @@ static void hloop_cleanup(hloop_t* loop) {
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    heap_init(&loop->realtimers, NULL);
 | 
					    heap_init(&loop->realtimers, NULL);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // signals
 | 
				
			||||||
 | 
					    printd("cleanup signals...\n");
 | 
				
			||||||
 | 
					    for (int i = 0; i < loop->signals.maxsize; ++i) {
 | 
				
			||||||
 | 
					        hsignal_t* sig = loop->signals.ptr[i];
 | 
				
			||||||
 | 
					        HV_FREE(sig);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    signal_array_cleanup(&loop->signals);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // readbuf
 | 
					    // readbuf
 | 
				
			||||||
    if (loop->readbuf.base && loop->readbuf.len) {
 | 
					    if (loop->readbuf.base && loop->readbuf.len) {
 | 
				
			||||||
        HV_FREE(loop->readbuf.base);
 | 
					        HV_FREE(loop->readbuf.base);
 | 
				
			||||||
| 
						 | 
					@ -587,6 +595,55 @@ void* hloop_userdata(hloop_t* loop) {
 | 
				
			||||||
    return loop->userdata;
 | 
					    return loop->userdata;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static hloop_t* s_signal_loop = NULL;
 | 
				
			||||||
 | 
					static void signal_handler(int signo) {
 | 
				
			||||||
 | 
					    if (!s_signal_loop) return;
 | 
				
			||||||
 | 
					    if (signo >= s_signal_loop->signals.maxsize) return;
 | 
				
			||||||
 | 
					    hsignal_t* sig = s_signal_loop->signals.ptr[signo];
 | 
				
			||||||
 | 
					    if (!sig) return;
 | 
				
			||||||
 | 
					    hloop_post_event(s_signal_loop, sig);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					hsignal_t* hsignal_add(hloop_t* loop, hsignal_cb cb, int signo) {
 | 
				
			||||||
 | 
					    int max_signo = 64;
 | 
				
			||||||
 | 
					#ifdef _NSIG
 | 
				
			||||||
 | 
					    max_signo = _NSIG;
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					    if (signo <= 0 || signo >= max_signo) {
 | 
				
			||||||
 | 
					        hloge("signo %d over %d!", signo, max_signo);
 | 
				
			||||||
 | 
					        return NULL;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (loop->signals.maxsize == 0) {
 | 
				
			||||||
 | 
					        signal_array_init(&loop->signals, max_signo);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    hsignal_t* sig = loop->signals.ptr[signo];
 | 
				
			||||||
 | 
					    if (sig == NULL) {
 | 
				
			||||||
 | 
					        HV_ALLOC_SIZEOF(sig);
 | 
				
			||||||
 | 
					        sig->loop = loop;
 | 
				
			||||||
 | 
					        sig->event_type = HEVENT_TYPE_SIGNAL;
 | 
				
			||||||
 | 
					        // NOTE: use event_id as signo
 | 
				
			||||||
 | 
					        sig->event_id = signo;
 | 
				
			||||||
 | 
					        sig->cb = cb;
 | 
				
			||||||
 | 
					        sig->priority = HEVENT_HIGHEST_PRIORITY;
 | 
				
			||||||
 | 
					        loop->signals.ptr[signo] = sig;
 | 
				
			||||||
 | 
					        loop->nsignals++;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    EVENT_ACTIVE(sig);
 | 
				
			||||||
 | 
					    s_signal_loop = loop;
 | 
				
			||||||
 | 
					    signal(signo, signal_handler);
 | 
				
			||||||
 | 
					    return sig;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void hsignal_del(hsignal_t* sig) {
 | 
				
			||||||
 | 
					    if (!sig->active) return;
 | 
				
			||||||
 | 
					    hloop_t* loop = sig->loop;
 | 
				
			||||||
 | 
					    int signo = (int)sig->event_id;
 | 
				
			||||||
 | 
					    if (signo >= loop->signals.maxsize) return;
 | 
				
			||||||
 | 
					    loop->signals.ptr[signo] = NULL;
 | 
				
			||||||
 | 
					    loop->nsignals--;
 | 
				
			||||||
 | 
					    EVENT_DEL(sig);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
hidle_t* hidle_add(hloop_t* loop, hidle_cb cb, uint32_t repeat) {
 | 
					hidle_t* hidle_add(hloop_t* loop, hidle_cb cb, uint32_t repeat) {
 | 
				
			||||||
    hidle_t* idle;
 | 
					    hidle_t* idle;
 | 
				
			||||||
    HV_ALLOC_SIZEOF(idle);
 | 
					    HV_ALLOC_SIZEOF(idle);
 | 
				
			||||||
| 
						 | 
					@ -1025,3 +1082,25 @@ hio_t* hloop_create_udp_server(hloop_t* loop, const char* host, int port) {
 | 
				
			||||||
hio_t* hloop_create_udp_client(hloop_t* loop, const char* host, int port) {
 | 
					hio_t* hloop_create_udp_client(hloop_t* loop, const char* host, int port) {
 | 
				
			||||||
    return hio_create_socket(loop, host, port, HIO_TYPE_UDP, HIO_CLIENT_SIDE);
 | 
					    return hio_create_socket(loop, host, port, HIO_TYPE_UDP, HIO_CLIENT_SIDE);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int hio_create_pipe(hloop_t* loop, hio_t* pipeio[2]) {
 | 
				
			||||||
 | 
					    int pipefd[2];
 | 
				
			||||||
 | 
					    hio_type_e type = HIO_TYPE_PIPE;
 | 
				
			||||||
 | 
					#if defined(OS_UNIX) && HAVE_PIPE
 | 
				
			||||||
 | 
					    if (pipe(pipefd) != 0) {
 | 
				
			||||||
 | 
					        hloge("pipe create failed!");
 | 
				
			||||||
 | 
					        return -1;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					#else
 | 
				
			||||||
 | 
					    if (Socketpair(AF_INET, SOCK_STREAM, 0, pipefd) != 0) {
 | 
				
			||||||
 | 
					        hloge("socketpair create failed!");
 | 
				
			||||||
 | 
					        return -1;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    type = HIO_TYPE_TCP;
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					    pipeio[0] = hio_get(loop, pipefd[0]);
 | 
				
			||||||
 | 
					    pipeio[1] = hio_get(loop, pipefd[1]);
 | 
				
			||||||
 | 
					    pipeio[0]->io_type = type;
 | 
				
			||||||
 | 
					    pipeio[1]->io_type = type;
 | 
				
			||||||
 | 
					    return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -11,16 +11,18 @@ typedef struct hevent_s     hevent_t;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// NOTE: The following structures are subclasses of hevent_t,
 | 
					// NOTE: The following structures are subclasses of hevent_t,
 | 
				
			||||||
// inheriting hevent_t data members and function members.
 | 
					// inheriting hevent_t data members and function members.
 | 
				
			||||||
 | 
					typedef struct hio_s        hio_t;
 | 
				
			||||||
typedef struct hidle_s      hidle_t;
 | 
					typedef struct hidle_s      hidle_t;
 | 
				
			||||||
typedef struct htimer_s     htimer_t;
 | 
					typedef struct htimer_s     htimer_t;
 | 
				
			||||||
typedef struct htimeout_s   htimeout_t;
 | 
					typedef struct htimeout_s   htimeout_t;
 | 
				
			||||||
typedef struct hperiod_s    hperiod_t;
 | 
					typedef struct hperiod_s    hperiod_t;
 | 
				
			||||||
typedef struct hio_s        hio_t;
 | 
					typedef struct hevent_s     hsignal_t;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
typedef void (*hevent_cb)   (hevent_t* ev);
 | 
					typedef void (*hevent_cb)   (hevent_t* ev);
 | 
				
			||||||
 | 
					typedef void (*hio_cb)      (hio_t* io);
 | 
				
			||||||
typedef void (*hidle_cb)    (hidle_t* idle);
 | 
					typedef void (*hidle_cb)    (hidle_t* idle);
 | 
				
			||||||
typedef void (*htimer_cb)   (htimer_t* timer);
 | 
					typedef void (*htimer_cb)   (htimer_t* timer);
 | 
				
			||||||
typedef void (*hio_cb)      (hio_t* io);
 | 
					typedef void (*hsignal_cb)  (hsignal_t* sig);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
typedef void (*haccept_cb)  (hio_t* io);
 | 
					typedef void (*haccept_cb)  (hio_t* io);
 | 
				
			||||||
typedef void (*hconnect_cb) (hio_t* io);
 | 
					typedef void (*hconnect_cb) (hio_t* io);
 | 
				
			||||||
| 
						 | 
					@ -42,6 +44,7 @@ typedef enum {
 | 
				
			||||||
    HEVENT_TYPE_PERIOD  = 0x00000020,
 | 
					    HEVENT_TYPE_PERIOD  = 0x00000020,
 | 
				
			||||||
    HEVENT_TYPE_TIMER   = HEVENT_TYPE_TIMEOUT|HEVENT_TYPE_PERIOD,
 | 
					    HEVENT_TYPE_TIMER   = HEVENT_TYPE_TIMEOUT|HEVENT_TYPE_PERIOD,
 | 
				
			||||||
    HEVENT_TYPE_IDLE    = 0x00000100,
 | 
					    HEVENT_TYPE_IDLE    = 0x00000100,
 | 
				
			||||||
 | 
					    HEVENT_TYPE_SIGNAL  = 0x00000200,
 | 
				
			||||||
    HEVENT_TYPE_CUSTOM  = 0x00000400, // 1024
 | 
					    HEVENT_TYPE_CUSTOM  = 0x00000400, // 1024
 | 
				
			||||||
} hevent_type_e;
 | 
					} hevent_type_e;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -94,6 +97,7 @@ typedef enum {
 | 
				
			||||||
    HIO_TYPE_STDIO      = 0x0000000F,
 | 
					    HIO_TYPE_STDIO      = 0x0000000F,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    HIO_TYPE_FILE       = 0x00000010,
 | 
					    HIO_TYPE_FILE       = 0x00000010,
 | 
				
			||||||
 | 
					    HIO_TYPE_PIPE       = 0x00000020,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    HIO_TYPE_IP         = 0x00000100,
 | 
					    HIO_TYPE_IP         = 0x00000100,
 | 
				
			||||||
    HIO_TYPE_SOCK_RAW   = 0x00000F00,
 | 
					    HIO_TYPE_SOCK_RAW   = 0x00000F00,
 | 
				
			||||||
| 
						 | 
					@ -183,6 +187,10 @@ HV_EXPORT void* hloop_userdata(hloop_t* loop);
 | 
				
			||||||
// NOTE: hloop_post_event is thread-safe, used to post event from other thread to loop thread.
 | 
					// NOTE: hloop_post_event is thread-safe, used to post event from other thread to loop thread.
 | 
				
			||||||
HV_EXPORT void hloop_post_event(hloop_t* loop, hevent_t* ev);
 | 
					HV_EXPORT void hloop_post_event(hloop_t* loop, hevent_t* ev);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// signal
 | 
				
			||||||
 | 
					HV_EXPORT hsignal_t* hsignal_add(hloop_t* loop, hsignal_cb cb, int signo);
 | 
				
			||||||
 | 
					HV_EXPORT void       hsignal_del(hsignal_t* sig);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// idle
 | 
					// idle
 | 
				
			||||||
HV_EXPORT hidle_t* hidle_add(hloop_t* loop, hidle_cb cb, uint32_t repeat DEFAULT(INFINITE));
 | 
					HV_EXPORT hidle_t* hidle_add(hloop_t* loop, hidle_cb cb, uint32_t repeat DEFAULT(INFINITE));
 | 
				
			||||||
HV_EXPORT void     hidle_del(hidle_t* idle);
 | 
					HV_EXPORT void     hidle_del(hidle_t* idle);
 | 
				
			||||||
| 
						 | 
					@ -436,6 +444,10 @@ HV_EXPORT hio_t* hloop_create_udp_server (hloop_t* loop, const char* host, int p
 | 
				
			||||||
// @see examples/nc.c
 | 
					// @see examples/nc.c
 | 
				
			||||||
HV_EXPORT hio_t* hloop_create_udp_client (hloop_t* loop, const char* host, int port);
 | 
					HV_EXPORT hio_t* hloop_create_udp_client (hloop_t* loop, const char* host, int port);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//-----------------pipe---------------------------------------------
 | 
				
			||||||
 | 
					// @see examples/pipe_test.c
 | 
				
			||||||
 | 
					HV_EXPORT int hio_create_pipe(hloop_t* loop, hio_t* pipeio[2]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//-----------------upstream---------------------------------------------
 | 
					//-----------------upstream---------------------------------------------
 | 
				
			||||||
// hio_read(io)
 | 
					// hio_read(io)
 | 
				
			||||||
// hio_read(io->upstream_io)
 | 
					// hio_read(io->upstream_io)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -287,7 +287,13 @@ static int __nio_write(hio_t* io, const void* buf, int len) {
 | 
				
			||||||
    case HIO_TYPE_UDP:
 | 
					    case HIO_TYPE_UDP:
 | 
				
			||||||
    case HIO_TYPE_KCP:
 | 
					    case HIO_TYPE_KCP:
 | 
				
			||||||
    case HIO_TYPE_IP:
 | 
					    case HIO_TYPE_IP:
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
        nwrite = sendto(io->fd, buf, len, 0, io->peeraddr, SOCKADDR_LEN(io->peeraddr));
 | 
					        nwrite = sendto(io->fd, buf, len, 0, io->peeraddr, SOCKADDR_LEN(io->peeraddr));
 | 
				
			||||||
 | 
					        if (((sockaddr_u*)io->localaddr)->sin.sin_port == 0) {
 | 
				
			||||||
 | 
					            socklen_t addrlen = sizeof(sockaddr_u);
 | 
				
			||||||
 | 
					            getsockname(io->fd, io->localaddr, &addrlen);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
        break;
 | 
					        break;
 | 
				
			||||||
    default:
 | 
					    default:
 | 
				
			||||||
        nwrite = write(io->fd, buf, len);
 | 
					        nwrite = write(io->fd, buf, len);
 | 
				
			||||||
| 
						 | 
					@ -317,15 +323,14 @@ read:
 | 
				
			||||||
            // goto read_done;
 | 
					            // goto read_done;
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
        } else if (err == EMSGSIZE) {
 | 
					        } else if (err == EMSGSIZE) {
 | 
				
			||||||
            // ignore
 | 
					            nread = len;
 | 
				
			||||||
            return;
 | 
					 | 
				
			||||||
        } else {
 | 
					        } else {
 | 
				
			||||||
            // perror("read");
 | 
					            // perror("read");
 | 
				
			||||||
            io->error = err;
 | 
					            io->error = err;
 | 
				
			||||||
            goto read_error;
 | 
					            goto read_error;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (nread == 0) {
 | 
					    if (nread == 0 && (io->io_type & HIO_TYPE_SOCK_STREAM)) {
 | 
				
			||||||
        goto disconnect;
 | 
					        goto disconnect;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (nread < len) {
 | 
					    if (nread < len) {
 | 
				
			||||||
| 
						 | 
					@ -379,7 +384,7 @@ write:
 | 
				
			||||||
            goto write_error;
 | 
					            goto write_error;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (nwrite == 0) {
 | 
					    if (nwrite == 0 && (io->io_type & HIO_TYPE_SOCK_STREAM)) {
 | 
				
			||||||
        goto disconnect;
 | 
					        goto disconnect;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    pbuf->offset += nwrite;
 | 
					    pbuf->offset += nwrite;
 | 
				
			||||||
| 
						 | 
					@ -510,12 +515,12 @@ try_write:
 | 
				
			||||||
                goto write_error;
 | 
					                goto write_error;
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        if (nwrite == 0) {
 | 
					 | 
				
			||||||
            goto disconnect;
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
        if (nwrite == len) {
 | 
					        if (nwrite == len) {
 | 
				
			||||||
            goto write_done;
 | 
					            goto write_done;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					        if (nwrite == 0 && (io->io_type & HIO_TYPE_SOCK_STREAM)) {
 | 
				
			||||||
 | 
					            goto disconnect;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
enqueue:
 | 
					enqueue:
 | 
				
			||||||
        hio_add(io, hio_handle_events, HV_WRITE);
 | 
					        hio_add(io, hio_handle_events, HV_WRITE);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
| 
						 | 
					@ -600,6 +605,8 @@ int hio_close (hio_t* io) {
 | 
				
			||||||
    SAFE_FREE(io->hostname);
 | 
					    SAFE_FREE(io->hostname);
 | 
				
			||||||
    if (io->io_type & HIO_TYPE_SOCKET) {
 | 
					    if (io->io_type & HIO_TYPE_SOCKET) {
 | 
				
			||||||
        closesocket(io->fd);
 | 
					        closesocket(io->fd);
 | 
				
			||||||
 | 
					    } else if (io->io_type == HIO_TYPE_PIPE) {
 | 
				
			||||||
 | 
					        close(io->fd);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    return 0;
 | 
					    return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -241,7 +241,7 @@ typedef std::shared_ptr<EventLoop> EventLoopPtr;
 | 
				
			||||||
static inline EventLoop* tlsEventLoop() {
 | 
					static inline EventLoop* tlsEventLoop() {
 | 
				
			||||||
    return (EventLoop*)ThreadLocalStorage::get(ThreadLocalStorage::EVENT_LOOP);
 | 
					    return (EventLoop*)ThreadLocalStorage::get(ThreadLocalStorage::EVENT_LOOP);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#define currentThreadEventLoop tlsEventLoop()
 | 
					#define currentThreadEventLoop ::hv::tlsEventLoop()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static inline TimerID setTimer(int timeout_ms, TimerCallback cb, uint32_t repeat = INFINITE) {
 | 
					static inline TimerID setTimer(int timeout_ms, TimerCallback cb, uint32_t repeat = INFINITE) {
 | 
				
			||||||
    EventLoop* loop = tlsEventLoop();
 | 
					    EventLoop* loop = tlsEventLoop();
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -44,6 +44,7 @@ public:
 | 
				
			||||||
               Functor pre = Functor(),
 | 
					               Functor pre = Functor(),
 | 
				
			||||||
               Functor post = Functor()) {
 | 
					               Functor post = Functor()) {
 | 
				
			||||||
        if (status() >= kStarting && status() < kStopped) return;
 | 
					        if (status() >= kStarting && status() < kStopped) return;
 | 
				
			||||||
 | 
					        if (isRunning()) return;
 | 
				
			||||||
        setStatus(kStarting);
 | 
					        setStatus(kStarting);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        thread_ = std::make_shared<std::thread>(&EventLoopThread::loop_thread, this, pre, post);
 | 
					        thread_ = std::make_shared<std::thread>(&EventLoopThread::loop_thread, this, pre, post);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -32,7 +32,11 @@ public:
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    EventLoopPtr loop(int idx = -1) {
 | 
					    EventLoopPtr loop(int idx = -1) {
 | 
				
			||||||
        return worker_threads.loop(idx);
 | 
					        EventLoopPtr worker_loop = worker_threads.loop(idx);
 | 
				
			||||||
 | 
					        if (worker_loop == NULL) {
 | 
				
			||||||
 | 
					            worker_loop = acceptor_loop;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        return worker_loop;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    //@retval >=0 listenfd, <0 error
 | 
					    //@retval >=0 listenfd, <0 error
 | 
				
			||||||
| 
						 | 
					@ -294,8 +298,10 @@ public:
 | 
				
			||||||
    // start thread-safe
 | 
					    // start thread-safe
 | 
				
			||||||
    void start(bool wait_threads_started = true) {
 | 
					    void start(bool wait_threads_started = true) {
 | 
				
			||||||
        TcpServerEventLoopTmpl<TSocketChannel>::start(wait_threads_started);
 | 
					        TcpServerEventLoopTmpl<TSocketChannel>::start(wait_threads_started);
 | 
				
			||||||
 | 
					        if (!isRunning()) {
 | 
				
			||||||
            EventLoopThread::start(wait_threads_started);
 | 
					            EventLoopThread::start(wait_threads_started);
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // stop thread-safe
 | 
					    // stop thread-safe
 | 
				
			||||||
    void stop(bool wait_threads_stopped = true) {
 | 
					    void stop(bool wait_threads_stopped = true) {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,6 +1,7 @@
 | 
				
			||||||
list(APPEND EXAMPLES
 | 
					list(APPEND EXAMPLES
 | 
				
			||||||
    hloop_test
 | 
					    hloop_test
 | 
				
			||||||
    htimer_test
 | 
					    htimer_test
 | 
				
			||||||
 | 
					    pipe_test
 | 
				
			||||||
    nc
 | 
					    nc
 | 
				
			||||||
    tinyhttpd
 | 
					    tinyhttpd
 | 
				
			||||||
    tinyproxyd
 | 
					    tinyproxyd
 | 
				
			||||||
| 
						 | 
					@ -23,6 +24,9 @@ target_link_libraries(hloop_test ${HV_LIBRARIES})
 | 
				
			||||||
add_executable(htimer_test htimer_test.c)
 | 
					add_executable(htimer_test htimer_test.c)
 | 
				
			||||||
target_link_libraries(htimer_test ${HV_LIBRARIES})
 | 
					target_link_libraries(htimer_test ${HV_LIBRARIES})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					add_executable(pipe_test pipe_test.c)
 | 
				
			||||||
 | 
					target_link_libraries(pipe_test ${HV_LIBRARIES})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
add_executable(nc nc.c)
 | 
					add_executable(nc nc.c)
 | 
				
			||||||
target_link_libraries(nc ${HV_LIBRARIES})
 | 
					target_link_libraries(nc ${HV_LIBRARIES})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -61,6 +65,22 @@ add_executable(jsonrpc_server jsonrpc/jsonrpc_server.c jsonrpc/cJSON.c)
 | 
				
			||||||
target_compile_definitions(jsonrpc_server PRIVATE CJSON_HIDE_SYMBOLS)
 | 
					target_compile_definitions(jsonrpc_server PRIVATE CJSON_HIDE_SYMBOLS)
 | 
				
			||||||
target_link_libraries(jsonrpc_server ${HV_LIBRARIES})
 | 
					target_link_libraries(jsonrpc_server ${HV_LIBRARIES})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(WITH_KCP)
 | 
				
			||||||
 | 
					    glob_headers_and_sources(KCPTUN_SMUX_FILES   kcptun/smux)
 | 
				
			||||||
 | 
					    glob_headers_and_sources(KCPTUN_CLIENT_FILES kcptun/client)
 | 
				
			||||||
 | 
					    glob_headers_and_sources(KCPTUN_SERVER_FILES kcptun/server)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # kcptun_client
 | 
				
			||||||
 | 
					    add_executable(kcptun_client ${KCPTUN_SMUX_FILES} ${KCPTUN_CLIENT_FILES})
 | 
				
			||||||
 | 
					    target_link_libraries(kcptun_client ${HV_LIBRARIES})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # kcptun_server
 | 
				
			||||||
 | 
					    add_executable(kcptun_server ${KCPTUN_SMUX_FILES} ${KCPTUN_SERVER_FILES})
 | 
				
			||||||
 | 
					    target_link_libraries(kcptun_server ${HV_LIBRARIES})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    list(APPEND EXAMPLES kcptun_client kcptun_server)
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if(WITH_EVPP)
 | 
					if(WITH_EVPP)
 | 
				
			||||||
    include_directories(../cpputil ../evpp)
 | 
					    include_directories(../cpputil ../evpp)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -5,6 +5,7 @@
 | 
				
			||||||
├── consul/                 consul服务注册与发现
 | 
					├── consul/                 consul服务注册与发现
 | 
				
			||||||
├── httpd/                  HTTP服务端
 | 
					├── httpd/                  HTTP服务端
 | 
				
			||||||
├── jsonrpc/                json RPC示例
 | 
					├── jsonrpc/                json RPC示例
 | 
				
			||||||
 | 
					├── kcptun/                 kcp隧道
 | 
				
			||||||
├── mqtt/                   MQTT发布订阅示例
 | 
					├── mqtt/                   MQTT发布订阅示例
 | 
				
			||||||
├── multi-thread/           多线程网络编程示例
 | 
					├── multi-thread/           多线程网络编程示例
 | 
				
			||||||
├── nmap/                   网络扫描工具
 | 
					├── nmap/                   网络扫描工具
 | 
				
			||||||
| 
						 | 
					@ -16,6 +17,7 @@
 | 
				
			||||||
├── http_client_test.c      HTTP客户端测试代码
 | 
					├── http_client_test.c      HTTP客户端测试代码
 | 
				
			||||||
├── http_server_test.c      HTTP服务端测试代码
 | 
					├── http_server_test.c      HTTP服务端测试代码
 | 
				
			||||||
├── nc.c                    网络连接工具
 | 
					├── nc.c                    网络连接工具
 | 
				
			||||||
 | 
					├── pipe_test.c             pipe示例代码
 | 
				
			||||||
├── socks5_proxy_server.c   SOCKS5代理服务
 | 
					├── socks5_proxy_server.c   SOCKS5代理服务
 | 
				
			||||||
├── tcp_chat_server.c       TCP聊天服务
 | 
					├── tcp_chat_server.c       TCP聊天服务
 | 
				
			||||||
├── tcp_echo_server.c       TCP回显服务
 | 
					├── tcp_echo_server.c       TCP回显服务
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -64,6 +64,11 @@ void on_custom_events(hevent_t* ev) {
 | 
				
			||||||
    printf("on_custom_events event_type=%d userdata=%ld\n", (int)ev->event_type, (long)(intptr_t)ev->userdata);
 | 
					    printf("on_custom_events event_type=%d userdata=%ld\n", (int)ev->event_type, (long)(intptr_t)ev->userdata);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void on_signal(hsignal_t* sig) {
 | 
				
			||||||
 | 
					    printf("on_signal signo=%d\n", (int)hevent_id(sig));
 | 
				
			||||||
 | 
					    hloop_stop(hevent_loop(sig));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int main() {
 | 
					int main() {
 | 
				
			||||||
    // memcheck atexit
 | 
					    // memcheck atexit
 | 
				
			||||||
    HV_MEMCHECK;
 | 
					    HV_MEMCHECK;
 | 
				
			||||||
| 
						 | 
					@ -87,6 +92,9 @@ int main() {
 | 
				
			||||||
    htimer_add_period(loop, cron_minutely, -1, -1, -1, -1, -1, INFINITE);
 | 
					    htimer_add_period(loop, cron_minutely, -1, -1, -1, -1, -1, INFINITE);
 | 
				
			||||||
    htimer_add_period(loop, cron_hourly, minute+1, -1, -1, -1, -1, INFINITE);
 | 
					    htimer_add_period(loop, cron_hourly, minute+1, -1, -1, -1, -1, INFINITE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // test signal: enter Ctrl-C to trigger
 | 
				
			||||||
 | 
					    hsignal_add(loop, on_signal, SIGINT);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // test network_logger
 | 
					    // test network_logger
 | 
				
			||||||
    htimer_add(loop, timer_write_log, 1000, INFINITE);
 | 
					    htimer_add(loop, timer_write_log, 1000, INFINITE);
 | 
				
			||||||
    hlog_set_handler(mylogger);
 | 
					    hlog_set_handler(mylogger);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -38,36 +38,25 @@ static void print_help();
 | 
				
			||||||
static int  parse_confile(const char* confile);
 | 
					static int  parse_confile(const char* confile);
 | 
				
			||||||
static void worker_fn(void* userdata);
 | 
					static void worker_fn(void* userdata);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// short options
 | 
					 | 
				
			||||||
static const char options[] = "hvc:ts:dp:";
 | 
					 | 
				
			||||||
// long options
 | 
					// long options
 | 
				
			||||||
static const option_t long_options[] = {
 | 
					static const option_t long_options[] = {
 | 
				
			||||||
    {'h', "help",       NO_ARGUMENT},
 | 
					    {'h', "help",       NO_ARGUMENT,        "Print this information"},
 | 
				
			||||||
    {'v', "version",    NO_ARGUMENT},
 | 
					    {'v', "version",    NO_ARGUMENT,        "Print version"},
 | 
				
			||||||
    {'c', "confile",    REQUIRED_ARGUMENT},
 | 
					    {'c', "confile",    REQUIRED_ARGUMENT,  "Set configure file, default etc/{program}.conf"},
 | 
				
			||||||
    {'t', "test",       NO_ARGUMENT},
 | 
					    {'t', "test",       NO_ARGUMENT,        "Test configure file and exit"},
 | 
				
			||||||
    {'s', "signal",     REQUIRED_ARGUMENT},
 | 
					    {'s', "signal",     REQUIRED_ARGUMENT,  "send signal to process, signal=[start,stop,restart,status,reload]"},
 | 
				
			||||||
    {'d', "daemon",     NO_ARGUMENT},
 | 
					    {'d', "daemon",     NO_ARGUMENT,        "Daemonize"},
 | 
				
			||||||
    {'p', "port",       REQUIRED_ARGUMENT}
 | 
					    {'p', "port",       REQUIRED_ARGUMENT,  "Set listen port"}
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
static const char detail_options[] = R"(
 | 
					 | 
				
			||||||
  -h|--help                 Print this information
 | 
					 | 
				
			||||||
  -v|--version              Print version
 | 
					 | 
				
			||||||
  -c|--confile <confile>    Set configure file, default etc/{program}.conf
 | 
					 | 
				
			||||||
  -t|--test                 Test configure file and exit
 | 
					 | 
				
			||||||
  -s|--signal <signal>      Send <signal> to process,
 | 
					 | 
				
			||||||
                            <signal>=[start,stop,restart,status,reload]
 | 
					 | 
				
			||||||
  -d|--daemon               Daemonize
 | 
					 | 
				
			||||||
  -p|--port <port>          Set listen port
 | 
					 | 
				
			||||||
)";
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
void print_version() {
 | 
					void print_version() {
 | 
				
			||||||
    printf("%s version %s\n", g_main_ctx.program_name, hv_compile_version());
 | 
					    printf("%s version %s\n", g_main_ctx.program_name, hv_compile_version());
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void print_help() {
 | 
					void print_help() {
 | 
				
			||||||
    printf("Usage: %s [%s]\n", g_main_ctx.program_name, options);
 | 
					    char detail_options[1024] = {0};
 | 
				
			||||||
    printf("Options:\n%s\n", detail_options);
 | 
					    dump_opt_long(long_options, ARRAY_SIZE(long_options), detail_options, sizeof(detail_options));
 | 
				
			||||||
 | 
					    printf("%s\n", detail_options);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int parse_confile(const char* confile) {
 | 
					int parse_confile(const char* confile) {
 | 
				
			||||||
| 
						 | 
					@ -170,7 +159,6 @@ int main(int argc, char** argv) {
 | 
				
			||||||
        print_help();
 | 
					        print_help();
 | 
				
			||||||
        exit(10);
 | 
					        exit(10);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    // int ret = parse_opt(argc, argv, options);
 | 
					 | 
				
			||||||
    int ret = parse_opt_long(argc, argv, long_options, ARRAY_SIZE(long_options));
 | 
					    int ret = parse_opt_long(argc, argv, long_options, ARRAY_SIZE(long_options));
 | 
				
			||||||
    if (ret != 0) {
 | 
					    if (ret != 0) {
 | 
				
			||||||
        print_help();
 | 
					        print_help();
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -16,36 +16,25 @@ static void print_help();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int  parse_confile(const char* confile);
 | 
					static int  parse_confile(const char* confile);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// short options
 | 
					 | 
				
			||||||
static const char options[] = "hvc:ts:dp:";
 | 
					 | 
				
			||||||
// long options
 | 
					// long options
 | 
				
			||||||
static const option_t long_options[] = {
 | 
					static const option_t long_options[] = {
 | 
				
			||||||
    {'h', "help",       NO_ARGUMENT},
 | 
					    {'h', "help",       NO_ARGUMENT,        "Print this information"},
 | 
				
			||||||
    {'v', "version",    NO_ARGUMENT},
 | 
					    {'v', "version",    NO_ARGUMENT,        "Print version"},
 | 
				
			||||||
    {'c', "confile",    REQUIRED_ARGUMENT},
 | 
					    {'c', "confile",    REQUIRED_ARGUMENT,  "Set configure file, default etc/{program}.conf"},
 | 
				
			||||||
    {'t', "test",       NO_ARGUMENT},
 | 
					    {'t', "test",       NO_ARGUMENT,        "Test configure file and exit"},
 | 
				
			||||||
    {'s', "signal",     REQUIRED_ARGUMENT},
 | 
					    {'s', "signal",     REQUIRED_ARGUMENT,  "send signal to process, signal=[start,stop,restart,status,reload]"},
 | 
				
			||||||
    {'d', "daemon",     NO_ARGUMENT},
 | 
					    {'d', "daemon",     NO_ARGUMENT,        "Daemonize"},
 | 
				
			||||||
    {'p', "port",       REQUIRED_ARGUMENT}
 | 
					    {'p', "port",       REQUIRED_ARGUMENT,  "Set listen port"}
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
static const char detail_options[] = R"(
 | 
					 | 
				
			||||||
  -h|--help                 Print this information
 | 
					 | 
				
			||||||
  -v|--version              Print version
 | 
					 | 
				
			||||||
  -c|--confile <confile>    Set configure file, default etc/{program}.conf
 | 
					 | 
				
			||||||
  -t|--test                 Test configure file and exit
 | 
					 | 
				
			||||||
  -s|--signal <signal>      Send <signal> to process,
 | 
					 | 
				
			||||||
                            <signal>=[start,stop,restart,status,reload]
 | 
					 | 
				
			||||||
  -d|--daemon               Daemonize
 | 
					 | 
				
			||||||
  -p|--port <port>          Set listen port
 | 
					 | 
				
			||||||
)";
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
void print_version() {
 | 
					void print_version() {
 | 
				
			||||||
    printf("%s version %s\n", g_main_ctx.program_name, hv_compile_version());
 | 
					    printf("%s version %s\n", g_main_ctx.program_name, hv_compile_version());
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void print_help() {
 | 
					void print_help() {
 | 
				
			||||||
    printf("Usage: %s [%s]\n", g_main_ctx.program_name, options);
 | 
					    char detail_options[1024] = {0};
 | 
				
			||||||
    printf("Options:\n%s\n", detail_options);
 | 
					    dump_opt_long(long_options, ARRAY_SIZE(long_options), detail_options, sizeof(detail_options));
 | 
				
			||||||
 | 
					    printf("%s\n", detail_options);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int parse_confile(const char* confile) {
 | 
					int parse_confile(const char* confile) {
 | 
				
			||||||
| 
						 | 
					@ -275,7 +264,6 @@ static void on_reload(void* userdata) {
 | 
				
			||||||
int main(int argc, char** argv) {
 | 
					int main(int argc, char** argv) {
 | 
				
			||||||
    // g_main_ctx
 | 
					    // g_main_ctx
 | 
				
			||||||
    main_ctx_init(argc, argv);
 | 
					    main_ctx_init(argc, argv);
 | 
				
			||||||
    //int ret = parse_opt(argc, argv, options);
 | 
					 | 
				
			||||||
    int ret = parse_opt_long(argc, argv, long_options, ARRAY_SIZE(long_options));
 | 
					    int ret = parse_opt_long(argc, argv, long_options, ARRAY_SIZE(long_options));
 | 
				
			||||||
    if (ret != 0) {
 | 
					    if (ret != 0) {
 | 
				
			||||||
        print_help();
 | 
					        print_help();
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -122,6 +122,7 @@ public:
 | 
				
			||||||
            calls_mutex.lock();
 | 
					            calls_mutex.lock();
 | 
				
			||||||
            auto iter = calls.find(res->id());
 | 
					            auto iter = calls.find(res->id());
 | 
				
			||||||
            if (iter == calls.end()) {
 | 
					            if (iter == calls.end()) {
 | 
				
			||||||
 | 
					                calls_mutex.unlock();
 | 
				
			||||||
                return;
 | 
					                return;
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            auto ctx = iter->second;
 | 
					            auto ctx = iter->second;
 | 
				
			||||||
| 
						 | 
					@ -147,7 +148,9 @@ public:
 | 
				
			||||||
        req->id();
 | 
					        req->id();
 | 
				
			||||||
        auto ctx = std::make_shared<protorpc::ProtoRpcContext>();
 | 
					        auto ctx = std::make_shared<protorpc::ProtoRpcContext>();
 | 
				
			||||||
        ctx->req = req;
 | 
					        ctx->req = req;
 | 
				
			||||||
 | 
					        calls_mutex.lock();
 | 
				
			||||||
        calls[req->id()] = ctx;
 | 
					        calls[req->id()] = ctx;
 | 
				
			||||||
 | 
					        calls_mutex.unlock();
 | 
				
			||||||
        // Request::SerializeToArray + protorpc_pack
 | 
					        // Request::SerializeToArray + protorpc_pack
 | 
				
			||||||
        protorpc_message msg;
 | 
					        protorpc_message msg;
 | 
				
			||||||
        protorpc_message_init(&msg);
 | 
					        protorpc_message_init(&msg);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -21,7 +21,7 @@ static int wget(const char* url, const char* filepath, wget_progress_cb progress
 | 
				
			||||||
    req.url = url;
 | 
					    req.url = url;
 | 
				
			||||||
    ret = cli.send(&req, &resp);
 | 
					    ret = cli.send(&req, &resp);
 | 
				
			||||||
    if (ret != 0) {
 | 
					    if (ret != 0) {
 | 
				
			||||||
        fprintf(stderr, "request error: %d\n", ret);
 | 
					        fprintf(stderr, "HEAD request error: %d\n", ret);
 | 
				
			||||||
        return ret;
 | 
					        return ret;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    printd("%s", resp.Dump(true, false).c_str());
 | 
					    printd("%s", resp.Dump(true, false).c_str());
 | 
				
			||||||
| 
						 | 
					@ -85,7 +85,7 @@ static int wget(const char* url, const char* filepath, wget_progress_cb progress
 | 
				
			||||||
        };
 | 
					        };
 | 
				
			||||||
        ret = cli.send(&req, &resp);
 | 
					        ret = cli.send(&req, &resp);
 | 
				
			||||||
        if (ret != 0) {
 | 
					        if (ret != 0) {
 | 
				
			||||||
            fprintf(stderr, "request error: %d\n", ret);
 | 
					            fprintf(stderr, "GET request error: %d\n", ret);
 | 
				
			||||||
            goto error;
 | 
					            goto error;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        goto success;
 | 
					        goto success;
 | 
				
			||||||
| 
						 | 
					@ -99,7 +99,7 @@ static int wget(const char* url, const char* filepath, wget_progress_cb progress
 | 
				
			||||||
        printd("%s", req.Dump(true, false).c_str());
 | 
					        printd("%s", req.Dump(true, false).c_str());
 | 
				
			||||||
        ret = cli.send(&req, &resp);
 | 
					        ret = cli.send(&req, &resp);
 | 
				
			||||||
        if (ret != 0) {
 | 
					        if (ret != 0) {
 | 
				
			||||||
            fprintf(stderr, "request error: %d\n", ret);
 | 
					            fprintf(stderr, "GET Range: bytes=%ld-%ld request error: %d\n", from, to, ret);
 | 
				
			||||||
            goto error;
 | 
					            goto error;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        printd("%s", resp.Dump(true, false).c_str());
 | 
					        printd("%s", resp.Dump(true, false).c_str());
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -37,32 +37,6 @@
 | 
				
			||||||
    #define HV_UNUSED
 | 
					    #define HV_UNUSED
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// @param[IN | OUT | INOUT]
 | 
					 | 
				
			||||||
#ifndef IN
 | 
					 | 
				
			||||||
#define IN
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#ifndef OUT
 | 
					 | 
				
			||||||
#define OUT
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#ifndef INOUT
 | 
					 | 
				
			||||||
#define INOUT
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// @field[OPTIONAL | REQUIRED | REPEATED]
 | 
					 | 
				
			||||||
#ifndef OPTIONAL
 | 
					 | 
				
			||||||
#define OPTIONAL
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#ifndef REQUIRED
 | 
					 | 
				
			||||||
#define REQUIRED
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#ifndef REPEATED
 | 
					 | 
				
			||||||
#define REPEATED
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#ifdef __cplusplus
 | 
					#ifdef __cplusplus
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifndef EXTERN_C
 | 
					#ifndef EXTERN_C
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -28,6 +28,9 @@ public:
 | 
				
			||||||
                HttpCookie cookie;
 | 
					                HttpCookie cookie;
 | 
				
			||||||
                if (cookie.parse(header_value)) {
 | 
					                if (cookie.parse(header_value)) {
 | 
				
			||||||
                    parsed->cookies.emplace_back(cookie);
 | 
					                    parsed->cookies.emplace_back(cookie);
 | 
				
			||||||
 | 
					                    header_field.clear();
 | 
				
			||||||
 | 
					                    header_value.clear();
 | 
				
			||||||
 | 
					                    return;
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
            parsed->headers[header_field] = header_value;
 | 
					            parsed->headers[header_field] = header_value;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -8,8 +8,28 @@
 | 
				
			||||||
#include "httpdef.h"    // import http_content_type_str_by_suffix
 | 
					#include "httpdef.h"    // import http_content_type_str_by_suffix
 | 
				
			||||||
#include "http_page.h"  // import make_index_of_page
 | 
					#include "http_page.h"  // import make_index_of_page
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef _MSC_VER
 | 
				
			||||||
 | 
					#include <codecvt>
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define ETAG_FMT    "\"%zx-%zx\""
 | 
					#define ETAG_FMT    "\"%zx-%zx\""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					FileCache::FileCache() {
 | 
				
			||||||
 | 
					    stat_interval = 10; // s
 | 
				
			||||||
 | 
					    expired_time  = 60; // s
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int hv_open(char const* filepath, int flags) {
 | 
				
			||||||
 | 
					#ifdef _MSC_VER
 | 
				
			||||||
 | 
					    std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>> conv;
 | 
				
			||||||
 | 
					    auto wfilepath = conv.from_bytes(filepath);
 | 
				
			||||||
 | 
					    int fd = _wopen(wfilepath.c_str(), flags);
 | 
				
			||||||
 | 
					#else
 | 
				
			||||||
 | 
					    int fd = open(filepath, flags);
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					    return fd;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
file_cache_ptr FileCache::Open(const char* filepath, OpenParam* param) {
 | 
					file_cache_ptr FileCache::Open(const char* filepath, OpenParam* param) {
 | 
				
			||||||
    std::lock_guard<std::mutex> locker(mutex_);
 | 
					    std::lock_guard<std::mutex> locker(mutex_);
 | 
				
			||||||
    file_cache_ptr fc = Get(filepath);
 | 
					    file_cache_ptr fc = Get(filepath);
 | 
				
			||||||
| 
						 | 
					@ -32,7 +52,7 @@ file_cache_ptr FileCache::Open(const char* filepath, OpenParam* param) {
 | 
				
			||||||
#ifdef O_BINARY
 | 
					#ifdef O_BINARY
 | 
				
			||||||
        flags |= O_BINARY;
 | 
					        flags |= O_BINARY;
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
        int fd = open(filepath, flags);
 | 
					        int fd = hv_open(filepath, flags);
 | 
				
			||||||
        if (fd < 0) {
 | 
					        if (fd < 0) {
 | 
				
			||||||
#ifdef OS_WIN
 | 
					#ifdef OS_WIN
 | 
				
			||||||
            // NOTE: open(dir) return -1 on windows
 | 
					            // NOTE: open(dir) return -1 on windows
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -64,10 +64,7 @@ public:
 | 
				
			||||||
    int             stat_interval;
 | 
					    int             stat_interval;
 | 
				
			||||||
    int             expired_time;
 | 
					    int             expired_time;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    FileCache() {
 | 
					    FileCache();
 | 
				
			||||||
        stat_interval = 10; // s
 | 
					 | 
				
			||||||
        expired_time  = 60; // s
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    struct OpenParam {
 | 
					    struct OpenParam {
 | 
				
			||||||
        bool need_read;
 | 
					        bool need_read;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -153,10 +153,11 @@ void HttpHandler::Close() {
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
bool HttpHandler::SwitchHTTP2() {
 | 
					bool HttpHandler::SwitchHTTP2() {
 | 
				
			||||||
    parser.reset(HttpParser::New(HTTP_SERVER, ::HTTP_V2));
 | 
					    HttpParser* http2_parser = HttpParser::New(HTTP_SERVER, ::HTTP_V2);
 | 
				
			||||||
    if (parser == NULL) {
 | 
					    if (http2_parser == NULL) {
 | 
				
			||||||
        return false;
 | 
					        return false;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					    parser.reset(http2_parser);
 | 
				
			||||||
    protocol = HTTP_V2;
 | 
					    protocol = HTTP_V2;
 | 
				
			||||||
    resp->http_major = req->http_major = 2;
 | 
					    resp->http_major = req->http_major = 2;
 | 
				
			||||||
    resp->http_minor = req->http_minor = 0;
 | 
					    resp->http_minor = req->http_minor = 0;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -68,7 +68,9 @@ int HttpResponseWriter::SSEvent(const std::string& data, const char* event /* =
 | 
				
			||||||
        EndHeaders("Content-Type", "text/event-stream");
 | 
					        EndHeaders("Content-Type", "text/event-stream");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    std::string msg;
 | 
					    std::string msg;
 | 
				
			||||||
 | 
					    if (event) {
 | 
				
			||||||
        msg = "event: "; msg += event; msg += "\n";
 | 
					        msg = "event: "; msg += event; msg += "\n";
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    msg += "data: ";  msg += data;  msg += "\n\n";
 | 
					    msg += "data: ";  msg += data;  msg += "\n\n";
 | 
				
			||||||
    state = SEND_BODY;
 | 
					    state = SEND_BODY;
 | 
				
			||||||
    return write(msg);
 | 
					    return write(msg);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -278,3 +278,34 @@ int http_server_stop(http_server_t* server) {
 | 
				
			||||||
    server->privdata = NULL;
 | 
					    server->privdata = NULL;
 | 
				
			||||||
    return 0;
 | 
					    return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace hv {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					std::shared_ptr<hv::EventLoop> HttpServer::loop(int idx) {
 | 
				
			||||||
 | 
					    HttpServerPrivdata* privdata = (HttpServerPrivdata*)this->privdata;
 | 
				
			||||||
 | 
					    if (privdata == NULL) return NULL;
 | 
				
			||||||
 | 
					    std::lock_guard<std::mutex> locker(privdata->mutex_);
 | 
				
			||||||
 | 
					    if (privdata->loops.empty()) return NULL;
 | 
				
			||||||
 | 
					    if (idx >= 0 && idx < (int)privdata->loops.size()) {
 | 
				
			||||||
 | 
					        return privdata->loops[idx];
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    EventLoop* cur = currentThreadEventLoop;
 | 
				
			||||||
 | 
					    for (auto& loop : privdata->loops) {
 | 
				
			||||||
 | 
					        if (loop.get() == cur) return loop;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return NULL;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					size_t HttpServer::connectionNum() {
 | 
				
			||||||
 | 
					    HttpServerPrivdata* privdata = (HttpServerPrivdata*)this->privdata;
 | 
				
			||||||
 | 
					    if (privdata == NULL) return 0;
 | 
				
			||||||
 | 
					    std::lock_guard<std::mutex> locker(privdata->mutex_);
 | 
				
			||||||
 | 
					    if (privdata->loops.empty()) return 0;
 | 
				
			||||||
 | 
					    size_t total = 0;
 | 
				
			||||||
 | 
					    for (auto& loop : privdata->loops) {
 | 
				
			||||||
 | 
					        total += loop->connectionNum;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return total;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,9 +3,11 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "hexport.h"
 | 
					#include "hexport.h"
 | 
				
			||||||
#include "hssl.h"
 | 
					#include "hssl.h"
 | 
				
			||||||
 | 
					// #include "EventLoop.h"
 | 
				
			||||||
#include "HttpService.h"
 | 
					#include "HttpService.h"
 | 
				
			||||||
// #include "WebSocketServer.h"
 | 
					// #include "WebSocketServer.h"
 | 
				
			||||||
namespace hv {
 | 
					namespace hv {
 | 
				
			||||||
 | 
					class EventLoop;
 | 
				
			||||||
struct WebSocketService;
 | 
					struct WebSocketService;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
using hv::HttpService;
 | 
					using hv::HttpService;
 | 
				
			||||||
| 
						 | 
					@ -81,7 +83,7 @@ int main() {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace hv {
 | 
					namespace hv {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class HttpServer : public http_server_t {
 | 
					class HV_EXPORT HttpServer : public http_server_t {
 | 
				
			||||||
public:
 | 
					public:
 | 
				
			||||||
    HttpServer(HttpService* service = NULL)
 | 
					    HttpServer(HttpService* service = NULL)
 | 
				
			||||||
        : http_server_t()
 | 
					        : http_server_t()
 | 
				
			||||||
| 
						 | 
					@ -94,6 +96,8 @@ public:
 | 
				
			||||||
        this->service = service;
 | 
					        this->service = service;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    std::shared_ptr<hv::EventLoop> loop(int idx = -1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    void setHost(const char* host = "0.0.0.0") {
 | 
					    void setHost(const char* host = "0.0.0.0") {
 | 
				
			||||||
        if (host) strcpy(this->host, host);
 | 
					        if (host) strcpy(this->host, host);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
| 
						 | 
					@ -115,6 +119,11 @@ public:
 | 
				
			||||||
        this->worker_threads = num;
 | 
					        this->worker_threads = num;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void setMaxWorkerConnectionNum(uint32_t num) {
 | 
				
			||||||
 | 
					        this->worker_connections = num;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    size_t connectionNum();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // SSL/TLS
 | 
					    // SSL/TLS
 | 
				
			||||||
    int setSslCtx(hssl_ctx_t ssl_ctx) {
 | 
					    int setSslCtx(hssl_ctx_t ssl_ctx) {
 | 
				
			||||||
        this->ssl_ctx = ssl_ctx;
 | 
					        this->ssl_ctx = ssl_ctx;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3,6 +3,7 @@
 | 
				
			||||||
#include "hlog.h"
 | 
					#include "hlog.h"
 | 
				
			||||||
#include "herr.h"
 | 
					#include "herr.h"
 | 
				
			||||||
#include "hendian.h"
 | 
					#include "hendian.h"
 | 
				
			||||||
 | 
					#include "hsocket.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static unsigned short mqtt_next_mid() {
 | 
					static unsigned short mqtt_next_mid() {
 | 
				
			||||||
    static unsigned short s_mid = 0;
 | 
					    static unsigned short s_mid = 0;
 | 
				
			||||||
| 
						 | 
					@ -165,10 +166,25 @@ static int mqtt_client_login(mqtt_client_t* cli) {
 | 
				
			||||||
    return nwrite < 0 ? nwrite : 0;
 | 
					    return nwrite < 0 ? nwrite : 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void connect_timeout_cb(htimer_t* timer) {
 | 
				
			||||||
 | 
					    mqtt_client_t* cli = (mqtt_client_t*)hevent_userdata(timer);
 | 
				
			||||||
 | 
					    if (cli == NULL) return;
 | 
				
			||||||
 | 
					    cli->timer = NULL;
 | 
				
			||||||
 | 
					    cli->error = ETIMEDOUT;
 | 
				
			||||||
 | 
					    hio_t* io = cli->io;
 | 
				
			||||||
 | 
					    if (io == NULL) return;
 | 
				
			||||||
 | 
					    char localaddrstr[SOCKADDR_STRLEN] = {0};
 | 
				
			||||||
 | 
					    char peeraddrstr[SOCKADDR_STRLEN] = {0};
 | 
				
			||||||
 | 
					    hlogw("connect timeout [%s] <=> [%s]",
 | 
				
			||||||
 | 
					            SOCKADDR_STR(hio_localaddr(io), localaddrstr),
 | 
				
			||||||
 | 
					            SOCKADDR_STR(hio_peeraddr(io), peeraddrstr));
 | 
				
			||||||
 | 
					    hio_close(io);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void reconnect_timer_cb(htimer_t* timer) {
 | 
					static void reconnect_timer_cb(htimer_t* timer) {
 | 
				
			||||||
    mqtt_client_t* cli = (mqtt_client_t*)hevent_userdata(timer);
 | 
					    mqtt_client_t* cli = (mqtt_client_t*)hevent_userdata(timer);
 | 
				
			||||||
    if (cli == NULL) return;
 | 
					    if (cli == NULL) return;
 | 
				
			||||||
    cli->reconn_timer = NULL;
 | 
					    cli->timer = NULL;
 | 
				
			||||||
    mqtt_client_reconnect(cli);
 | 
					    mqtt_client_reconnect(cli);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -182,8 +198,8 @@ static void on_close(hio_t* io) {
 | 
				
			||||||
    // reconnect
 | 
					    // reconnect
 | 
				
			||||||
    if (cli->reconn_setting && reconn_setting_can_retry(cli->reconn_setting)) {
 | 
					    if (cli->reconn_setting && reconn_setting_can_retry(cli->reconn_setting)) {
 | 
				
			||||||
        uint32_t delay = reconn_setting_calc_delay(cli->reconn_setting);
 | 
					        uint32_t delay = reconn_setting_calc_delay(cli->reconn_setting);
 | 
				
			||||||
        cli->reconn_timer = htimer_add(cli->loop, reconnect_timer_cb, delay, 1);
 | 
					        cli->timer = htimer_add(cli->loop, reconnect_timer_cb, delay, 1);
 | 
				
			||||||
        hevent_set_userdata(cli->reconn_timer, cli);
 | 
					        hevent_set_userdata(cli->timer, cli);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -214,6 +230,10 @@ static void on_packet(hio_t* io, void* buf, int len) {
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        cli->connected = 1;
 | 
					        cli->connected = 1;
 | 
				
			||||||
 | 
					        if (cli->timer) {
 | 
				
			||||||
 | 
					            htimer_del(cli->timer);
 | 
				
			||||||
 | 
					            cli->timer = NULL;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
        if (cli->keepalive) {
 | 
					        if (cli->keepalive) {
 | 
				
			||||||
            cli->ping_cnt = 0;
 | 
					            cli->ping_cnt = 0;
 | 
				
			||||||
            hio_set_heartbeat(io, cli->keepalive * 1000, mqtt_send_ping);
 | 
					            hio_set_heartbeat(io, cli->keepalive * 1000, mqtt_send_ping);
 | 
				
			||||||
| 
						 | 
					@ -465,6 +485,12 @@ void mqtt_client_set_connect_timeout(mqtt_client_t* cli, int ms) {
 | 
				
			||||||
    cli->connect_timeout = ms;
 | 
					    cli->connect_timeout = ms;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void mqtt_client_set_host(mqtt_client_t* cli, const char* host, int port, int ssl) {
 | 
				
			||||||
 | 
					    hv_strncpy(cli->host, host, sizeof(cli->host));
 | 
				
			||||||
 | 
					    cli->port = port;
 | 
				
			||||||
 | 
					    cli->ssl = ssl;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int mqtt_client_connect(mqtt_client_t* cli, const char* host, int port, int ssl) {
 | 
					int mqtt_client_connect(mqtt_client_t* cli, const char* host, int port, int ssl) {
 | 
				
			||||||
    if (!cli) return -1;
 | 
					    if (!cli) return -1;
 | 
				
			||||||
    hv_strncpy(cli->host, host, sizeof(cli->host));
 | 
					    hv_strncpy(cli->host, host, sizeof(cli->host));
 | 
				
			||||||
| 
						 | 
					@ -478,13 +504,14 @@ int mqtt_client_connect(mqtt_client_t* cli, const char* host, int port, int ssl)
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        hio_enable_ssl(io);
 | 
					        hio_enable_ssl(io);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    if (cli->connect_timeout > 0) {
 | 
					 | 
				
			||||||
        hio_set_connect_timeout(io, cli->connect_timeout);
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    cli->io = io;
 | 
					    cli->io = io;
 | 
				
			||||||
    hevent_set_userdata(io, cli);
 | 
					    hevent_set_userdata(io, cli);
 | 
				
			||||||
    hio_setcb_connect(io, on_connect);
 | 
					    hio_setcb_connect(io, on_connect);
 | 
				
			||||||
    hio_setcb_close(io, on_close);
 | 
					    hio_setcb_close(io, on_close);
 | 
				
			||||||
 | 
					    if (cli->connect_timeout > 0) {
 | 
				
			||||||
 | 
					        cli->timer = htimer_add(cli->loop, connect_timeout_cb, cli->connect_timeout, 1);
 | 
				
			||||||
 | 
					        hevent_set_userdata(cli->timer, cli);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    return hio_connect(io);
 | 
					    return hio_connect(io);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -48,7 +48,7 @@ struct mqtt_client_s {
 | 
				
			||||||
    // privdata
 | 
					    // privdata
 | 
				
			||||||
    hloop_t*    loop;
 | 
					    hloop_t*    loop;
 | 
				
			||||||
    hio_t*      io;
 | 
					    hio_t*      io;
 | 
				
			||||||
    htimer_t*   reconn_timer;
 | 
					    htimer_t*   timer;
 | 
				
			||||||
    // SSL/TLS
 | 
					    // SSL/TLS
 | 
				
			||||||
    hssl_ctx_t ssl_ctx;
 | 
					    hssl_ctx_t ssl_ctx;
 | 
				
			||||||
    // thread-safe
 | 
					    // thread-safe
 | 
				
			||||||
| 
						 | 
					@ -102,6 +102,7 @@ HV_EXPORT int mqtt_client_reconnect(mqtt_client_t* cli);
 | 
				
			||||||
// on_connect -> mqtt_client_login ->
 | 
					// on_connect -> mqtt_client_login ->
 | 
				
			||||||
// on_connack
 | 
					// on_connack
 | 
				
			||||||
HV_EXPORT void mqtt_client_set_connect_timeout(mqtt_client_t* cli, int ms);
 | 
					HV_EXPORT void mqtt_client_set_connect_timeout(mqtt_client_t* cli, int ms);
 | 
				
			||||||
 | 
					HV_EXPORT void mqtt_client_set_host(mqtt_client_t* cli, const char* host, int port, int ssl);
 | 
				
			||||||
HV_EXPORT int  mqtt_client_connect(mqtt_client_t* cli,
 | 
					HV_EXPORT int  mqtt_client_connect(mqtt_client_t* cli,
 | 
				
			||||||
        const char* host,
 | 
					        const char* host,
 | 
				
			||||||
        int port DEFAULT(DEFAULT_MQTT_PORT),
 | 
					        int port DEFAULT(DEFAULT_MQTT_PORT),
 | 
				
			||||||
| 
						 | 
					@ -203,6 +204,10 @@ public:
 | 
				
			||||||
        mqtt_client_set_connect_timeout(client, ms);
 | 
					        mqtt_client_set_connect_timeout(client, ms);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void setHost(const char* host, int port = DEFAULT_MQTT_PORT, int ssl = 0) {
 | 
				
			||||||
 | 
					        mqtt_client_set_host(client, host, port, ssl);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    int connect(const char* host, int port = DEFAULT_MQTT_PORT, int ssl = 0) {
 | 
					    int connect(const char* host, int port = DEFAULT_MQTT_PORT, int ssl = 0) {
 | 
				
			||||||
        return mqtt_client_connect(client, host, port, ssl);
 | 
					        return mqtt_client_connect(client, host, port, ssl);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -30,6 +30,7 @@ int main(int argc, char* argv[]) {
 | 
				
			||||||
            4);
 | 
					            4);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const char* test_urls[] = {
 | 
					    const char* test_urls[] = {
 | 
				
			||||||
 | 
					        "http://user:pswd@[www.example.com]:80/path?query#fragment",
 | 
				
			||||||
        "http://user:pswd@www.example.com:80/path?query#fragment",
 | 
					        "http://user:pswd@www.example.com:80/path?query#fragment",
 | 
				
			||||||
        "http://user:pswd@www.example.com/path?query#fragment",
 | 
					        "http://user:pswd@www.example.com/path?query#fragment",
 | 
				
			||||||
        "http://www.example.com/path?query#fragment",
 | 
					        "http://www.example.com/path?query#fragment",
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -4,7 +4,7 @@
 | 
				
			||||||
#include "hexport.h"
 | 
					#include "hexport.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define BASE64_ENCODE_OUT_SIZE(s)   (((s) + 2) / 3 * 4)
 | 
					#define BASE64_ENCODE_OUT_SIZE(s)   (((s) + 2) / 3 * 4)
 | 
				
			||||||
#define BASE64_DECODE_OUT_SIZE(s)   (((s)) / 4 * 3)
 | 
					#define BASE64_DECODE_OUT_SIZE(s)   (((s) + 3) / 4 * 3)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
BEGIN_EXTERN_C
 | 
					BEGIN_EXTERN_C
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -36,7 +36,11 @@ HV_INLINE std::string Base64Decode(const char* str, unsigned int len = 0) {
 | 
				
			||||||
    int decoded_size = BASE64_DECODE_OUT_SIZE(len);
 | 
					    int decoded_size = BASE64_DECODE_OUT_SIZE(len);
 | 
				
			||||||
    std::string decoded_buf(decoded_size + 1, 0);
 | 
					    std::string decoded_buf(decoded_size + 1, 0);
 | 
				
			||||||
    decoded_size = hv_base64_decode(str, len, (unsigned char*)decoded_buf.data());
 | 
					    decoded_size = hv_base64_decode(str, len, (unsigned char*)decoded_buf.data());
 | 
				
			||||||
 | 
					    if (decoded_size > 0) {
 | 
				
			||||||
        decoded_buf.resize(decoded_size);
 | 
					        decoded_buf.resize(decoded_size);
 | 
				
			||||||
 | 
					    } else {
 | 
				
			||||||
 | 
					        decoded_buf.clear();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    return decoded_buf;
 | 
					    return decoded_buf;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,218 @@
 | 
				
			||||||
 | 
					cmake_minimum_required(VERSION 2.4.4...3.15.0)
 | 
				
			||||||
 | 
					set(CMAKE_ALLOW_LOOSE_LOOP_CONSTRUCTS ON)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					project(zlib C)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					set(VERSION "1.3.1")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					option(ZLIB_BUILD_EXAMPLES "Enable Zlib Examples" ON)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					set(INSTALL_BIN_DIR "${CMAKE_INSTALL_PREFIX}/bin" CACHE PATH "Installation directory for executables")
 | 
				
			||||||
 | 
					set(INSTALL_LIB_DIR "${CMAKE_INSTALL_PREFIX}/lib" CACHE PATH "Installation directory for libraries")
 | 
				
			||||||
 | 
					set(INSTALL_INC_DIR "${CMAKE_INSTALL_PREFIX}/include" CACHE PATH "Installation directory for headers")
 | 
				
			||||||
 | 
					set(INSTALL_MAN_DIR "${CMAKE_INSTALL_PREFIX}/share/man" CACHE PATH "Installation directory for manual pages")
 | 
				
			||||||
 | 
					set(INSTALL_PKGCONFIG_DIR "${CMAKE_INSTALL_PREFIX}/share/pkgconfig" CACHE PATH "Installation directory for pkgconfig (.pc) files")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					include(CheckTypeSize)
 | 
				
			||||||
 | 
					include(CheckFunctionExists)
 | 
				
			||||||
 | 
					include(CheckIncludeFile)
 | 
				
			||||||
 | 
					include(CheckCSourceCompiles)
 | 
				
			||||||
 | 
					enable_testing()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					check_include_file(sys/types.h HAVE_SYS_TYPES_H)
 | 
				
			||||||
 | 
					check_include_file(stdint.h    HAVE_STDINT_H)
 | 
				
			||||||
 | 
					check_include_file(stddef.h    HAVE_STDDEF_H)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# Check to see if we have large file support
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					set(CMAKE_REQUIRED_DEFINITIONS -D_LARGEFILE64_SOURCE=1)
 | 
				
			||||||
 | 
					# We add these other definitions here because CheckTypeSize.cmake
 | 
				
			||||||
 | 
					# in CMake 2.4.x does not automatically do so and we want
 | 
				
			||||||
 | 
					# compatibility with CMake 2.4.x.
 | 
				
			||||||
 | 
					if(HAVE_SYS_TYPES_H)
 | 
				
			||||||
 | 
					    list(APPEND CMAKE_REQUIRED_DEFINITIONS -DHAVE_SYS_TYPES_H)
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					if(HAVE_STDINT_H)
 | 
				
			||||||
 | 
					    list(APPEND CMAKE_REQUIRED_DEFINITIONS -DHAVE_STDINT_H)
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					if(HAVE_STDDEF_H)
 | 
				
			||||||
 | 
					    list(APPEND CMAKE_REQUIRED_DEFINITIONS -DHAVE_STDDEF_H)
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					check_type_size(off64_t OFF64_T)
 | 
				
			||||||
 | 
					if(HAVE_OFF64_T)
 | 
				
			||||||
 | 
					   add_definitions(-D_LARGEFILE64_SOURCE=1)
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					set(CMAKE_REQUIRED_DEFINITIONS) # clear variable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# Check for fseeko
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					check_function_exists(fseeko HAVE_FSEEKO)
 | 
				
			||||||
 | 
					if(NOT HAVE_FSEEKO)
 | 
				
			||||||
 | 
					    add_definitions(-DNO_FSEEKO)
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# Check for unistd.h
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					check_include_file(unistd.h Z_HAVE_UNISTD_H)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(MSVC)
 | 
				
			||||||
 | 
					    set(CMAKE_DEBUG_POSTFIX "d")
 | 
				
			||||||
 | 
					    add_definitions(-D_CRT_SECURE_NO_DEPRECATE)
 | 
				
			||||||
 | 
					    add_definitions(-D_CRT_NONSTDC_NO_DEPRECATE)
 | 
				
			||||||
 | 
					    include_directories(${CMAKE_CURRENT_SOURCE_DIR})
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(NOT CMAKE_CURRENT_SOURCE_DIR STREQUAL CMAKE_CURRENT_BINARY_DIR)
 | 
				
			||||||
 | 
					    # If we're doing an out of source build and the user has a zconf.h
 | 
				
			||||||
 | 
					    # in their source tree...
 | 
				
			||||||
 | 
					    if(EXISTS ${CMAKE_CURRENT_SOURCE_DIR}/zconf.h)
 | 
				
			||||||
 | 
					        message(STATUS "Renaming")
 | 
				
			||||||
 | 
					        message(STATUS "    ${CMAKE_CURRENT_SOURCE_DIR}/zconf.h")
 | 
				
			||||||
 | 
					        message(STATUS "to 'zconf.h.included' because this file is included with zlib")
 | 
				
			||||||
 | 
					        message(STATUS "but CMake generates it automatically in the build directory.")
 | 
				
			||||||
 | 
					        file(RENAME ${CMAKE_CURRENT_SOURCE_DIR}/zconf.h ${CMAKE_CURRENT_SOURCE_DIR}/zconf.h.included)
 | 
				
			||||||
 | 
					  endif()
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					set(ZLIB_PC ${CMAKE_CURRENT_BINARY_DIR}/zlib.pc)
 | 
				
			||||||
 | 
					configure_file( ${CMAKE_CURRENT_SOURCE_DIR}/zlib.pc.cmakein
 | 
				
			||||||
 | 
							${ZLIB_PC} @ONLY)
 | 
				
			||||||
 | 
					configure_file(	${CMAKE_CURRENT_SOURCE_DIR}/zconf.h.cmakein
 | 
				
			||||||
 | 
							${CMAKE_CURRENT_BINARY_DIR}/zconf.h @ONLY)
 | 
				
			||||||
 | 
					include_directories(${CMAKE_CURRENT_BINARY_DIR} ${CMAKE_SOURCE_DIR})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#============================================================================
 | 
				
			||||||
 | 
					# zlib
 | 
				
			||||||
 | 
					#============================================================================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					set(ZLIB_PUBLIC_HDRS
 | 
				
			||||||
 | 
					    ${CMAKE_CURRENT_BINARY_DIR}/zconf.h
 | 
				
			||||||
 | 
					    zlib.h
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					set(ZLIB_PRIVATE_HDRS
 | 
				
			||||||
 | 
					    crc32.h
 | 
				
			||||||
 | 
					    deflate.h
 | 
				
			||||||
 | 
					    gzguts.h
 | 
				
			||||||
 | 
					    inffast.h
 | 
				
			||||||
 | 
					    inffixed.h
 | 
				
			||||||
 | 
					    inflate.h
 | 
				
			||||||
 | 
					    inftrees.h
 | 
				
			||||||
 | 
					    trees.h
 | 
				
			||||||
 | 
					    zutil.h
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					set(ZLIB_SRCS
 | 
				
			||||||
 | 
					    adler32.c
 | 
				
			||||||
 | 
					    compress.c
 | 
				
			||||||
 | 
					    crc32.c
 | 
				
			||||||
 | 
					    deflate.c
 | 
				
			||||||
 | 
					    gzclose.c
 | 
				
			||||||
 | 
					    gzlib.c
 | 
				
			||||||
 | 
					    gzread.c
 | 
				
			||||||
 | 
					    gzwrite.c
 | 
				
			||||||
 | 
					    inflate.c
 | 
				
			||||||
 | 
					    infback.c
 | 
				
			||||||
 | 
					    inftrees.c
 | 
				
			||||||
 | 
					    inffast.c
 | 
				
			||||||
 | 
					    trees.c
 | 
				
			||||||
 | 
					    uncompr.c
 | 
				
			||||||
 | 
					    zutil.c
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(NOT MINGW)
 | 
				
			||||||
 | 
					    set(ZLIB_DLL_SRCS
 | 
				
			||||||
 | 
					        win32/zlib1.rc # If present will override custom build rule below.
 | 
				
			||||||
 | 
					    )
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# parse the full version number from zlib.h and include in ZLIB_FULL_VERSION
 | 
				
			||||||
 | 
					file(READ ${CMAKE_CURRENT_SOURCE_DIR}/zlib.h _zlib_h_contents)
 | 
				
			||||||
 | 
					string(REGEX REPLACE ".*#define[ \t]+ZLIB_VERSION[ \t]+\"([-0-9A-Za-z.]+)\".*"
 | 
				
			||||||
 | 
					    "\\1" ZLIB_FULL_VERSION ${_zlib_h_contents})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(MINGW)
 | 
				
			||||||
 | 
					    # This gets us DLL resource information when compiling on MinGW.
 | 
				
			||||||
 | 
					    if(NOT CMAKE_RC_COMPILER)
 | 
				
			||||||
 | 
					        set(CMAKE_RC_COMPILER windres.exe)
 | 
				
			||||||
 | 
					    endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    add_custom_command(OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/zlib1rc.obj
 | 
				
			||||||
 | 
					                       COMMAND ${CMAKE_RC_COMPILER}
 | 
				
			||||||
 | 
					                            -D GCC_WINDRES
 | 
				
			||||||
 | 
					                            -I ${CMAKE_CURRENT_SOURCE_DIR}
 | 
				
			||||||
 | 
					                            -I ${CMAKE_CURRENT_BINARY_DIR}
 | 
				
			||||||
 | 
					                            -o ${CMAKE_CURRENT_BINARY_DIR}/zlib1rc.obj
 | 
				
			||||||
 | 
					                            -i ${CMAKE_CURRENT_SOURCE_DIR}/win32/zlib1.rc)
 | 
				
			||||||
 | 
					    set(ZLIB_DLL_SRCS ${CMAKE_CURRENT_BINARY_DIR}/zlib1rc.obj)
 | 
				
			||||||
 | 
					endif(MINGW)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					add_library(zlib SHARED ${ZLIB_SRCS} ${ZLIB_DLL_SRCS} ${ZLIB_PUBLIC_HDRS} ${ZLIB_PRIVATE_HDRS})
 | 
				
			||||||
 | 
					target_include_directories(zlib PUBLIC ${CMAKE_CURRENT_BINARY_DIR} ${CMAKE_CURRENT_SOURCE_DIR})
 | 
				
			||||||
 | 
					add_library(zlibstatic STATIC ${ZLIB_SRCS} ${ZLIB_PUBLIC_HDRS} ${ZLIB_PRIVATE_HDRS})
 | 
				
			||||||
 | 
					target_include_directories(zlibstatic PUBLIC ${CMAKE_CURRENT_BINARY_DIR} ${CMAKE_CURRENT_SOURCE_DIR})
 | 
				
			||||||
 | 
					set_target_properties(zlib PROPERTIES DEFINE_SYMBOL ZLIB_DLL)
 | 
				
			||||||
 | 
					set_target_properties(zlib PROPERTIES SOVERSION 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(NOT CYGWIN)
 | 
				
			||||||
 | 
					    # This property causes shared libraries on Linux to have the full version
 | 
				
			||||||
 | 
					    # encoded into their final filename.  We disable this on Cygwin because
 | 
				
			||||||
 | 
					    # it causes cygz-${ZLIB_FULL_VERSION}.dll to be created when cygz.dll
 | 
				
			||||||
 | 
					    # seems to be the default.
 | 
				
			||||||
 | 
					    #
 | 
				
			||||||
 | 
					    # This has no effect with MSVC, on that platform the version info for
 | 
				
			||||||
 | 
					    # the DLL comes from the resource file win32/zlib1.rc
 | 
				
			||||||
 | 
					    set_target_properties(zlib PROPERTIES VERSION ${ZLIB_FULL_VERSION})
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(UNIX)
 | 
				
			||||||
 | 
					    # On unix-like platforms the library is almost always called libz
 | 
				
			||||||
 | 
					   set_target_properties(zlib zlibstatic PROPERTIES OUTPUT_NAME z)
 | 
				
			||||||
 | 
					   if(NOT APPLE AND NOT(CMAKE_SYSTEM_NAME STREQUAL AIX))
 | 
				
			||||||
 | 
					     set_target_properties(zlib PROPERTIES LINK_FLAGS "-Wl,--version-script,\"${CMAKE_CURRENT_SOURCE_DIR}/zlib.map\"")
 | 
				
			||||||
 | 
					   endif()
 | 
				
			||||||
 | 
					elseif(BUILD_SHARED_LIBS AND WIN32)
 | 
				
			||||||
 | 
					    # Creates zlib1.dll when building shared library version
 | 
				
			||||||
 | 
					    set_target_properties(zlib PROPERTIES SUFFIX "1.dll")
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if(NOT SKIP_INSTALL_LIBRARIES AND NOT SKIP_INSTALL_ALL )
 | 
				
			||||||
 | 
					    install(TARGETS zlib zlibstatic
 | 
				
			||||||
 | 
					        RUNTIME DESTINATION "${INSTALL_BIN_DIR}"
 | 
				
			||||||
 | 
					        ARCHIVE DESTINATION "${INSTALL_LIB_DIR}"
 | 
				
			||||||
 | 
					        LIBRARY DESTINATION "${INSTALL_LIB_DIR}" )
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					if(NOT SKIP_INSTALL_HEADERS AND NOT SKIP_INSTALL_ALL )
 | 
				
			||||||
 | 
					    install(FILES ${ZLIB_PUBLIC_HDRS} DESTINATION "${INSTALL_INC_DIR}")
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					if(NOT SKIP_INSTALL_FILES AND NOT SKIP_INSTALL_ALL )
 | 
				
			||||||
 | 
					    install(FILES zlib.3 DESTINATION "${INSTALL_MAN_DIR}/man3")
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					if(NOT SKIP_INSTALL_FILES AND NOT SKIP_INSTALL_ALL )
 | 
				
			||||||
 | 
					    install(FILES ${ZLIB_PC} DESTINATION "${INSTALL_PKGCONFIG_DIR}")
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#============================================================================
 | 
				
			||||||
 | 
					# Example binaries
 | 
				
			||||||
 | 
					#============================================================================
 | 
				
			||||||
 | 
					if(ZLIB_BUILD_EXAMPLES)
 | 
				
			||||||
 | 
					    add_executable(example test/example.c)
 | 
				
			||||||
 | 
					    target_link_libraries(example zlib)
 | 
				
			||||||
 | 
					    add_test(example example)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    add_executable(minigzip test/minigzip.c)
 | 
				
			||||||
 | 
					    target_link_libraries(minigzip zlib)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if(HAVE_OFF64_T)
 | 
				
			||||||
 | 
					        add_executable(example64 test/example.c)
 | 
				
			||||||
 | 
					        target_link_libraries(example64 zlib)
 | 
				
			||||||
 | 
					        set_target_properties(example64 PROPERTIES COMPILE_FLAGS "-D_FILE_OFFSET_BITS=64")
 | 
				
			||||||
 | 
					        add_test(example64 example64)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        add_executable(minigzip64 test/minigzip.c)
 | 
				
			||||||
 | 
					        target_link_libraries(minigzip64 zlib)
 | 
				
			||||||
 | 
					        set_target_properties(minigzip64 PROPERTIES COMPILE_FLAGS "-D_FILE_OFFSET_BITS=64")
 | 
				
			||||||
 | 
					    endif()
 | 
				
			||||||
 | 
					endif()
 | 
				
			||||||
											
												
													File diff suppressed because it is too large
													Load Diff
												
											
										
									
								| 
						 | 
					@ -0,0 +1,367 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                Frequently Asked Questions about zlib
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					If your question is not there, please check the zlib home page
 | 
				
			||||||
 | 
					http://zlib.net/ which may have more recent information.
 | 
				
			||||||
 | 
					The latest zlib FAQ is at http://zlib.net/zlib_faq.html
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 1. Is zlib Y2K-compliant?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Yes. zlib doesn't handle dates.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 2. Where can I get a Windows DLL version?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    The zlib sources can be compiled without change to produce a DLL.  See the
 | 
				
			||||||
 | 
					    file win32/DLL_FAQ.txt in the zlib distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 3. Where can I get a Visual Basic interface to zlib?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    See
 | 
				
			||||||
 | 
					        * http://marknelson.us/1997/01/01/zlib-engine/
 | 
				
			||||||
 | 
					        * win32/DLL_FAQ.txt in the zlib distribution
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 4. compress() returns Z_BUF_ERROR.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Make sure that before the call of compress(), the length of the compressed
 | 
				
			||||||
 | 
					    buffer is equal to the available size of the compressed buffer and not
 | 
				
			||||||
 | 
					    zero.  For Visual Basic, check that this parameter is passed by reference
 | 
				
			||||||
 | 
					    ("as any"), not by value ("as long").
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 5. deflate() or inflate() returns Z_BUF_ERROR.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Before making the call, make sure that avail_in and avail_out are not zero.
 | 
				
			||||||
 | 
					    When setting the parameter flush equal to Z_FINISH, also make sure that
 | 
				
			||||||
 | 
					    avail_out is big enough to allow processing all pending input.  Note that a
 | 
				
			||||||
 | 
					    Z_BUF_ERROR is not fatal--another call to deflate() or inflate() can be
 | 
				
			||||||
 | 
					    made with more input or output space.  A Z_BUF_ERROR may in fact be
 | 
				
			||||||
 | 
					    unavoidable depending on how the functions are used, since it is not
 | 
				
			||||||
 | 
					    possible to tell whether or not there is more output pending when
 | 
				
			||||||
 | 
					    strm.avail_out returns with zero.  See http://zlib.net/zlib_how.html for a
 | 
				
			||||||
 | 
					    heavily annotated example.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 6. Where's the zlib documentation (man pages, etc.)?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    It's in zlib.h .  Examples of zlib usage are in the files test/example.c
 | 
				
			||||||
 | 
					    and test/minigzip.c, with more in examples/ .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 7. Why don't you use GNU autoconf or libtool or ...?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Because we would like to keep zlib as a very small and simple package.
 | 
				
			||||||
 | 
					    zlib is rather portable and doesn't need much configuration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 8. I found a bug in zlib.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Most of the time, such problems are due to an incorrect usage of zlib.
 | 
				
			||||||
 | 
					    Please try to reproduce the problem with a small program and send the
 | 
				
			||||||
 | 
					    corresponding source to us at zlib@gzip.org .  Do not send multi-megabyte
 | 
				
			||||||
 | 
					    data files without prior agreement.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 9. Why do I get "undefined reference to gzputc"?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    If "make test" produces something like
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					       example.o(.text+0x154): undefined reference to `gzputc'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    check that you don't have old files libz.* in /usr/lib, /usr/local/lib or
 | 
				
			||||||
 | 
					    /usr/X11R6/lib. Remove any old versions, then do "make install".
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					10. I need a Delphi interface to zlib.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    See the contrib/delphi directory in the zlib distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					11. Can zlib handle .zip archives?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Not by itself, no.  See the directory contrib/minizip in the zlib
 | 
				
			||||||
 | 
					    distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					12. Can zlib handle .Z files?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No, sorry.  You have to spawn an uncompress or gunzip subprocess, or adapt
 | 
				
			||||||
 | 
					    the code of uncompress on your own.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					13. How can I make a Unix shared library?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    By default a shared (and a static) library is built for Unix.  So:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    make distclean
 | 
				
			||||||
 | 
					    ./configure
 | 
				
			||||||
 | 
					    make
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					14. How do I install a shared zlib library on Unix?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    After the above, then:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    make install
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    However, many flavors of Unix come with a shared zlib already installed.
 | 
				
			||||||
 | 
					    Before going to the trouble of compiling a shared version of zlib and
 | 
				
			||||||
 | 
					    trying to install it, you may want to check if it's already there!  If you
 | 
				
			||||||
 | 
					    can #include <zlib.h>, it's there.  The -lz option will probably link to
 | 
				
			||||||
 | 
					    it.  You can check the version at the top of zlib.h or with the
 | 
				
			||||||
 | 
					    ZLIB_VERSION symbol defined in zlib.h .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					15. I have a question about OttoPDF.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    We are not the authors of OttoPDF. The real author is on the OttoPDF web
 | 
				
			||||||
 | 
					    site: Joel Hainley, jhainley@myndkryme.com.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					16. Can zlib decode Flate data in an Adobe PDF file?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Yes. See http://www.pdflib.com/ . To modify PDF forms, see
 | 
				
			||||||
 | 
					    http://sourceforge.net/projects/acroformtool/ .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					17. Why am I getting this "register_frame_info not found" error on Solaris?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    After installing zlib 1.1.4 on Solaris 2.6, running applications using zlib
 | 
				
			||||||
 | 
					    generates an error such as:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ld.so.1: rpm: fatal: relocation error: file /usr/local/lib/libz.so:
 | 
				
			||||||
 | 
					        symbol __register_frame_info: referenced symbol not found
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    The symbol __register_frame_info is not part of zlib, it is generated by
 | 
				
			||||||
 | 
					    the C compiler (cc or gcc).  You must recompile applications using zlib
 | 
				
			||||||
 | 
					    which have this problem.  This problem is specific to Solaris.  See
 | 
				
			||||||
 | 
					    http://www.sunfreeware.com for Solaris versions of zlib and applications
 | 
				
			||||||
 | 
					    using zlib.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					18. Why does gzip give an error on a file I make with compress/deflate?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    The compress and deflate functions produce data in the zlib format, which
 | 
				
			||||||
 | 
					    is different and incompatible with the gzip format.  The gz* functions in
 | 
				
			||||||
 | 
					    zlib on the other hand use the gzip format.  Both the zlib and gzip formats
 | 
				
			||||||
 | 
					    use the same compressed data format internally, but have different headers
 | 
				
			||||||
 | 
					    and trailers around the compressed data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					19. Ok, so why are there two different formats?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    The gzip format was designed to retain the directory information about a
 | 
				
			||||||
 | 
					    single file, such as the name and last modification date.  The zlib format
 | 
				
			||||||
 | 
					    on the other hand was designed for in-memory and communication channel
 | 
				
			||||||
 | 
					    applications, and has a much more compact header and trailer and uses a
 | 
				
			||||||
 | 
					    faster integrity check than gzip.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					20. Well that's nice, but how do I make a gzip file in memory?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    You can request that deflate write the gzip format instead of the zlib
 | 
				
			||||||
 | 
					    format using deflateInit2().  You can also request that inflate decode the
 | 
				
			||||||
 | 
					    gzip format using inflateInit2().  Read zlib.h for more details.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					21. Is zlib thread-safe?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Yes.  However any library routines that zlib uses and any application-
 | 
				
			||||||
 | 
					    provided memory allocation routines must also be thread-safe.  zlib's gz*
 | 
				
			||||||
 | 
					    functions use stdio library routines, and most of zlib's functions use the
 | 
				
			||||||
 | 
					    library memory allocation routines by default.  zlib's *Init* functions
 | 
				
			||||||
 | 
					    allow for the application to provide custom memory allocation routines.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Of course, you should only operate on any given zlib or gzip stream from a
 | 
				
			||||||
 | 
					    single thread at a time.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					22. Can I use zlib in my commercial application?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Yes.  Please read the license in zlib.h.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					23. Is zlib under the GNU license?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No.  Please read the license in zlib.h.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					24. The license says that altered source versions must be "plainly marked". So
 | 
				
			||||||
 | 
					    what exactly do I need to do to meet that requirement?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    You need to change the ZLIB_VERSION and ZLIB_VERNUM #defines in zlib.h.  In
 | 
				
			||||||
 | 
					    particular, the final version number needs to be changed to "f", and an
 | 
				
			||||||
 | 
					    identification string should be appended to ZLIB_VERSION.  Version numbers
 | 
				
			||||||
 | 
					    x.x.x.f are reserved for modifications to zlib by others than the zlib
 | 
				
			||||||
 | 
					    maintainers.  For example, if the version of the base zlib you are altering
 | 
				
			||||||
 | 
					    is "1.2.3.4", then in zlib.h you should change ZLIB_VERNUM to 0x123f, and
 | 
				
			||||||
 | 
					    ZLIB_VERSION to something like "1.2.3.f-zachary-mods-v3".  You can also
 | 
				
			||||||
 | 
					    update the version strings in deflate.c and inftrees.c.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    For altered source distributions, you should also note the origin and
 | 
				
			||||||
 | 
					    nature of the changes in zlib.h, as well as in ChangeLog and README, along
 | 
				
			||||||
 | 
					    with the dates of the alterations.  The origin should include at least your
 | 
				
			||||||
 | 
					    name (or your company's name), and an email address to contact for help or
 | 
				
			||||||
 | 
					    issues with the library.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Note that distributing a compiled zlib library along with zlib.h and
 | 
				
			||||||
 | 
					    zconf.h is also a source distribution, and so you should change
 | 
				
			||||||
 | 
					    ZLIB_VERSION and ZLIB_VERNUM and note the origin and nature of the changes
 | 
				
			||||||
 | 
					    in zlib.h as you would for a full source distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					25. Will zlib work on a big-endian or little-endian architecture, and can I
 | 
				
			||||||
 | 
					    exchange compressed data between them?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Yes and yes.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					26. Will zlib work on a 64-bit machine?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Yes.  It has been tested on 64-bit machines, and has no dependence on any
 | 
				
			||||||
 | 
					    data types being limited to 32-bits in length.  If you have any
 | 
				
			||||||
 | 
					    difficulties, please provide a complete problem report to zlib@gzip.org
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					27. Will zlib decompress data from the PKWare Data Compression Library?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No.  The PKWare DCL uses a completely different compressed data format than
 | 
				
			||||||
 | 
					    does PKZIP and zlib.  However, you can look in zlib's contrib/blast
 | 
				
			||||||
 | 
					    directory for a possible solution to your problem.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					28. Can I access data randomly in a compressed stream?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No, not without some preparation.  If when compressing you periodically use
 | 
				
			||||||
 | 
					    Z_FULL_FLUSH, carefully write all the pending data at those points, and
 | 
				
			||||||
 | 
					    keep an index of those locations, then you can start decompression at those
 | 
				
			||||||
 | 
					    points.  You have to be careful to not use Z_FULL_FLUSH too often, since it
 | 
				
			||||||
 | 
					    can significantly degrade compression.  Alternatively, you can scan a
 | 
				
			||||||
 | 
					    deflate stream once to generate an index, and then use that index for
 | 
				
			||||||
 | 
					    random access.  See examples/zran.c .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					29. Does zlib work on MVS, OS/390, CICS, etc.?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    It has in the past, but we have not heard of any recent evidence.  There
 | 
				
			||||||
 | 
					    were working ports of zlib 1.1.4 to MVS, but those links no longer work.
 | 
				
			||||||
 | 
					    If you know of recent, successful applications of zlib on these operating
 | 
				
			||||||
 | 
					    systems, please let us know.  Thanks.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					30. Is there some simpler, easier to read version of inflate I can look at to
 | 
				
			||||||
 | 
					    understand the deflate format?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    First off, you should read RFC 1951.  Second, yes.  Look in zlib's
 | 
				
			||||||
 | 
					    contrib/puff directory.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					31. Does zlib infringe on any patents?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    As far as we know, no.  In fact, that was originally the whole point behind
 | 
				
			||||||
 | 
					    zlib.  Look here for some more information:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    http://www.gzip.org/#faq11
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					32. Can zlib work with greater than 4 GB of data?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Yes.  inflate() and deflate() will process any amount of data correctly.
 | 
				
			||||||
 | 
					    Each call of inflate() or deflate() is limited to input and output chunks
 | 
				
			||||||
 | 
					    of the maximum value that can be stored in the compiler's "unsigned int"
 | 
				
			||||||
 | 
					    type, but there is no limit to the number of chunks.  Note however that the
 | 
				
			||||||
 | 
					    strm.total_in and strm_total_out counters may be limited to 4 GB.  These
 | 
				
			||||||
 | 
					    counters are provided as a convenience and are not used internally by
 | 
				
			||||||
 | 
					    inflate() or deflate().  The application can easily set up its own counters
 | 
				
			||||||
 | 
					    updated after each call of inflate() or deflate() to count beyond 4 GB.
 | 
				
			||||||
 | 
					    compress() and uncompress() may be limited to 4 GB, since they operate in a
 | 
				
			||||||
 | 
					    single call.  gzseek() and gztell() may be limited to 4 GB depending on how
 | 
				
			||||||
 | 
					    zlib is compiled.  See the zlibCompileFlags() function in zlib.h.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    The word "may" appears several times above since there is a 4 GB limit only
 | 
				
			||||||
 | 
					    if the compiler's "long" type is 32 bits.  If the compiler's "long" type is
 | 
				
			||||||
 | 
					    64 bits, then the limit is 16 exabytes.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					33. Does zlib have any security vulnerabilities?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    The only one that we are aware of is potentially in gzprintf().  If zlib is
 | 
				
			||||||
 | 
					    compiled to use sprintf() or vsprintf(), then there is no protection
 | 
				
			||||||
 | 
					    against a buffer overflow of an 8K string space (or other value as set by
 | 
				
			||||||
 | 
					    gzbuffer()), other than the caller of gzprintf() assuring that the output
 | 
				
			||||||
 | 
					    will not exceed 8K.  On the other hand, if zlib is compiled to use
 | 
				
			||||||
 | 
					    snprintf() or vsnprintf(), which should normally be the case, then there is
 | 
				
			||||||
 | 
					    no vulnerability.  The ./configure script will display warnings if an
 | 
				
			||||||
 | 
					    insecure variation of sprintf() will be used by gzprintf().  Also the
 | 
				
			||||||
 | 
					    zlibCompileFlags() function will return information on what variant of
 | 
				
			||||||
 | 
					    sprintf() is used by gzprintf().
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    If you don't have snprintf() or vsnprintf() and would like one, you can
 | 
				
			||||||
 | 
					    find a portable implementation here:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        http://www.ijs.si/software/snprintf/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Note that you should be using the most recent version of zlib.  Versions
 | 
				
			||||||
 | 
					    1.1.3 and before were subject to a double-free vulnerability, and versions
 | 
				
			||||||
 | 
					    1.2.1 and 1.2.2 were subject to an access exception when decompressing
 | 
				
			||||||
 | 
					    invalid compressed data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					34. Is there a Java version of zlib?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Probably what you want is to use zlib in Java. zlib is already included
 | 
				
			||||||
 | 
					    as part of the Java SDK in the java.util.zip package. If you really want
 | 
				
			||||||
 | 
					    a version of zlib written in the Java language, look on the zlib home
 | 
				
			||||||
 | 
					    page for links: http://zlib.net/ .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					35. I get this or that compiler or source-code scanner warning when I crank it
 | 
				
			||||||
 | 
					    up to maximally-pedantic. Can't you guys write proper code?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Many years ago, we gave up attempting to avoid warnings on every compiler
 | 
				
			||||||
 | 
					    in the universe.  It just got to be a waste of time, and some compilers
 | 
				
			||||||
 | 
					    were downright silly as well as contradicted each other.  So now, we simply
 | 
				
			||||||
 | 
					    make sure that the code always works.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					36. Valgrind (or some similar memory access checker) says that deflate is
 | 
				
			||||||
 | 
					    performing a conditional jump that depends on an uninitialized value.
 | 
				
			||||||
 | 
					    Isn't that a bug?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No.  That is intentional for performance reasons, and the output of deflate
 | 
				
			||||||
 | 
					    is not affected.  This only started showing up recently since zlib 1.2.x
 | 
				
			||||||
 | 
					    uses malloc() by default for allocations, whereas earlier versions used
 | 
				
			||||||
 | 
					    calloc(), which zeros out the allocated memory.  Even though the code was
 | 
				
			||||||
 | 
					    correct, versions 1.2.4 and later was changed to not stimulate these
 | 
				
			||||||
 | 
					    checkers.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					37. Will zlib read the (insert any ancient or arcane format here) compressed
 | 
				
			||||||
 | 
					    data format?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Probably not. Look in the comp.compression FAQ for pointers to various
 | 
				
			||||||
 | 
					    formats and associated software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					38. How can I encrypt/decrypt zip files with zlib?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    zlib doesn't support encryption.  The original PKZIP encryption is very
 | 
				
			||||||
 | 
					    weak and can be broken with freely available programs.  To get strong
 | 
				
			||||||
 | 
					    encryption, use GnuPG, http://www.gnupg.org/ , which already includes zlib
 | 
				
			||||||
 | 
					    compression.  For PKZIP compatible "encryption", look at
 | 
				
			||||||
 | 
					    http://www.info-zip.org/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					39. What's the difference between the "gzip" and "deflate" HTTP 1.1 encodings?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    "gzip" is the gzip format, and "deflate" is the zlib format.  They should
 | 
				
			||||||
 | 
					    probably have called the second one "zlib" instead to avoid confusion with
 | 
				
			||||||
 | 
					    the raw deflate compressed data format.  While the HTTP 1.1 RFC 2616
 | 
				
			||||||
 | 
					    correctly points to the zlib specification in RFC 1950 for the "deflate"
 | 
				
			||||||
 | 
					    transfer encoding, there have been reports of servers and browsers that
 | 
				
			||||||
 | 
					    incorrectly produce or expect raw deflate data per the deflate
 | 
				
			||||||
 | 
					    specification in RFC 1951, most notably Microsoft.  So even though the
 | 
				
			||||||
 | 
					    "deflate" transfer encoding using the zlib format would be the more
 | 
				
			||||||
 | 
					    efficient approach (and in fact exactly what the zlib format was designed
 | 
				
			||||||
 | 
					    for), using the "gzip" transfer encoding is probably more reliable due to
 | 
				
			||||||
 | 
					    an unfortunate choice of name on the part of the HTTP 1.1 authors.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Bottom line: use the gzip format for HTTP 1.1 encoding.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					40. Does zlib support the new "Deflate64" format introduced by PKWare?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No.  PKWare has apparently decided to keep that format proprietary, since
 | 
				
			||||||
 | 
					    they have not documented it as they have previous compression formats.  In
 | 
				
			||||||
 | 
					    any case, the compression improvements are so modest compared to other more
 | 
				
			||||||
 | 
					    modern approaches, that it's not worth the effort to implement.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					41. I'm having a problem with the zip functions in zlib, can you help?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    There are no zip functions in zlib.  You are probably using minizip by
 | 
				
			||||||
 | 
					    Giles Vollant, which is found in the contrib directory of zlib.  It is not
 | 
				
			||||||
 | 
					    part of zlib.  In fact none of the stuff in contrib is part of zlib.  The
 | 
				
			||||||
 | 
					    files in there are not supported by the zlib authors.  You need to contact
 | 
				
			||||||
 | 
					    the authors of the respective contribution for help.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					42. The match.asm code in contrib is under the GNU General Public License.
 | 
				
			||||||
 | 
					    Since it's part of zlib, doesn't that mean that all of zlib falls under the
 | 
				
			||||||
 | 
					    GNU GPL?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No.  The files in contrib are not part of zlib.  They were contributed by
 | 
				
			||||||
 | 
					    other authors and are provided as a convenience to the user within the zlib
 | 
				
			||||||
 | 
					    distribution.  Each item in contrib has its own license.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					43. Is zlib subject to export controls?  What is its ECCN?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    zlib is not subject to export controls, and so is classified as EAR99.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					44. Can you please sign these lengthy legal documents and fax them back to us
 | 
				
			||||||
 | 
					    so that we can use your software in our product?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    No. Go away. Shoo.
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,68 @@
 | 
				
			||||||
 | 
					CMakeLists.txt  cmake build file
 | 
				
			||||||
 | 
					ChangeLog       history of changes
 | 
				
			||||||
 | 
					FAQ             Frequently Asked Questions about zlib
 | 
				
			||||||
 | 
					INDEX           this file
 | 
				
			||||||
 | 
					Makefile        dummy Makefile that tells you to ./configure
 | 
				
			||||||
 | 
					Makefile.in     template for Unix Makefile
 | 
				
			||||||
 | 
					README          guess what
 | 
				
			||||||
 | 
					configure       configure script for Unix
 | 
				
			||||||
 | 
					make_vms.com    makefile for VMS
 | 
				
			||||||
 | 
					test/example.c  zlib usages examples for build testing
 | 
				
			||||||
 | 
					test/minigzip.c minimal gzip-like functionality for build testing
 | 
				
			||||||
 | 
					test/infcover.c inf*.c code coverage for build coverage testing
 | 
				
			||||||
 | 
					treebuild.xml   XML description of source file dependencies
 | 
				
			||||||
 | 
					zconf.h.cmakein zconf.h template for cmake
 | 
				
			||||||
 | 
					zconf.h.in      zconf.h template for configure
 | 
				
			||||||
 | 
					zlib.3          Man page for zlib
 | 
				
			||||||
 | 
					zlib.3.pdf      Man page in PDF format
 | 
				
			||||||
 | 
					zlib.map        Linux symbol information
 | 
				
			||||||
 | 
					zlib.pc.in      Template for pkg-config descriptor
 | 
				
			||||||
 | 
					zlib.pc.cmakein zlib.pc template for cmake
 | 
				
			||||||
 | 
					zlib2ansi       perl script to convert source files for C++ compilation
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					amiga/          makefiles for Amiga SAS C
 | 
				
			||||||
 | 
					as400/          makefiles for AS/400
 | 
				
			||||||
 | 
					doc/            documentation for formats and algorithms
 | 
				
			||||||
 | 
					msdos/          makefiles for MSDOS
 | 
				
			||||||
 | 
					nintendods/     makefile for Nintendo DS
 | 
				
			||||||
 | 
					old/            makefiles for various architectures and zlib documentation
 | 
				
			||||||
 | 
					                files that have not yet been updated for zlib 1.2.x
 | 
				
			||||||
 | 
					qnx/            makefiles for QNX
 | 
				
			||||||
 | 
					watcom/         makefiles for OpenWatcom
 | 
				
			||||||
 | 
					win32/          makefiles for Windows
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                zlib public header files (required for library use):
 | 
				
			||||||
 | 
					zconf.h
 | 
				
			||||||
 | 
					zlib.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                private source files used to build the zlib library:
 | 
				
			||||||
 | 
					adler32.c
 | 
				
			||||||
 | 
					compress.c
 | 
				
			||||||
 | 
					crc32.c
 | 
				
			||||||
 | 
					crc32.h
 | 
				
			||||||
 | 
					deflate.c
 | 
				
			||||||
 | 
					deflate.h
 | 
				
			||||||
 | 
					gzclose.c
 | 
				
			||||||
 | 
					gzguts.h
 | 
				
			||||||
 | 
					gzlib.c
 | 
				
			||||||
 | 
					gzread.c
 | 
				
			||||||
 | 
					gzwrite.c
 | 
				
			||||||
 | 
					infback.c
 | 
				
			||||||
 | 
					inffast.c
 | 
				
			||||||
 | 
					inffast.h
 | 
				
			||||||
 | 
					inffixed.h
 | 
				
			||||||
 | 
					inflate.c
 | 
				
			||||||
 | 
					inflate.h
 | 
				
			||||||
 | 
					inftrees.c
 | 
				
			||||||
 | 
					inftrees.h
 | 
				
			||||||
 | 
					trees.c
 | 
				
			||||||
 | 
					trees.h
 | 
				
			||||||
 | 
					uncompr.c
 | 
				
			||||||
 | 
					zutil.c
 | 
				
			||||||
 | 
					zutil.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                source files for sample programs
 | 
				
			||||||
 | 
					See examples/README.examples
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                unsupported contributions by third parties
 | 
				
			||||||
 | 
					See contrib/README.contrib
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,22 @@
 | 
				
			||||||
 | 
					Copyright notice:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 (C) 1995-2022 Jean-loup Gailly and Mark Adler
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  This software is provided 'as-is', without any express or implied
 | 
				
			||||||
 | 
					  warranty.  In no event will the authors be held liable for any damages
 | 
				
			||||||
 | 
					  arising from the use of this software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Permission is granted to anyone to use this software for any purpose,
 | 
				
			||||||
 | 
					  including commercial applications, and to alter it and redistribute it
 | 
				
			||||||
 | 
					  freely, subject to the following restrictions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  1. The origin of this software must not be misrepresented; you must not
 | 
				
			||||||
 | 
					     claim that you wrote the original software. If you use this software
 | 
				
			||||||
 | 
					     in a product, an acknowledgment in the product documentation would be
 | 
				
			||||||
 | 
					     appreciated but is not required.
 | 
				
			||||||
 | 
					  2. Altered source versions must be plainly marked as such, and must not be
 | 
				
			||||||
 | 
					     misrepresented as being the original software.
 | 
				
			||||||
 | 
					  3. This notice may not be removed or altered from any source distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Jean-loup Gailly        Mark Adler
 | 
				
			||||||
 | 
					  jloup@gzip.org          madler@alumni.caltech.edu
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,5 @@
 | 
				
			||||||
 | 
					all:
 | 
				
			||||||
 | 
						-@echo "Please use ./configure first.  Thank you."
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					distclean:
 | 
				
			||||||
 | 
						make -f Makefile.in distclean
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,410 @@
 | 
				
			||||||
 | 
					# Makefile for zlib
 | 
				
			||||||
 | 
					# Copyright (C) 1995-2024 Jean-loup Gailly, Mark Adler
 | 
				
			||||||
 | 
					# For conditions of distribution and use, see copyright notice in zlib.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# To compile and test, type:
 | 
				
			||||||
 | 
					#    ./configure; make test
 | 
				
			||||||
 | 
					# Normally configure builds both a static and a shared library.
 | 
				
			||||||
 | 
					# If you want to build just a static library, use: ./configure --static
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# To install /usr/local/lib/libz.* and /usr/local/include/zlib.h, type:
 | 
				
			||||||
 | 
					#    make install
 | 
				
			||||||
 | 
					# To install in $HOME instead of /usr/local, use:
 | 
				
			||||||
 | 
					#    make install prefix=$HOME
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CC=cc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CFLAGS=-O
 | 
				
			||||||
 | 
					#CFLAGS=-O -DMAX_WBITS=14 -DMAX_MEM_LEVEL=7
 | 
				
			||||||
 | 
					#CFLAGS=-g -DZLIB_DEBUG
 | 
				
			||||||
 | 
					#CFLAGS=-O3 -Wall -Wwrite-strings -Wpointer-arith -Wconversion \
 | 
				
			||||||
 | 
					#           -Wstrict-prototypes -Wmissing-prototypes
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					SFLAGS=-O
 | 
				
			||||||
 | 
					LDFLAGS=
 | 
				
			||||||
 | 
					TEST_LIBS=-L. libz.a
 | 
				
			||||||
 | 
					LDSHARED=$(CC)
 | 
				
			||||||
 | 
					CPP=$(CC) -E
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					STATICLIB=libz.a
 | 
				
			||||||
 | 
					SHAREDLIB=libz.so
 | 
				
			||||||
 | 
					SHAREDLIBV=libz.so.1.3.1
 | 
				
			||||||
 | 
					SHAREDLIBM=libz.so.1
 | 
				
			||||||
 | 
					LIBS=$(STATICLIB) $(SHAREDLIBV)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					AR=ar
 | 
				
			||||||
 | 
					ARFLAGS=rc
 | 
				
			||||||
 | 
					RANLIB=ranlib
 | 
				
			||||||
 | 
					LDCONFIG=ldconfig
 | 
				
			||||||
 | 
					LDSHAREDLIBC=-lc
 | 
				
			||||||
 | 
					TAR=tar
 | 
				
			||||||
 | 
					SHELL=/bin/sh
 | 
				
			||||||
 | 
					EXE=
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					prefix = /usr/local
 | 
				
			||||||
 | 
					exec_prefix = ${prefix}
 | 
				
			||||||
 | 
					libdir = ${exec_prefix}/lib
 | 
				
			||||||
 | 
					sharedlibdir = ${libdir}
 | 
				
			||||||
 | 
					includedir = ${prefix}/include
 | 
				
			||||||
 | 
					mandir = ${prefix}/share/man
 | 
				
			||||||
 | 
					man3dir = ${mandir}/man3
 | 
				
			||||||
 | 
					pkgconfigdir = ${libdir}/pkgconfig
 | 
				
			||||||
 | 
					SRCDIR=
 | 
				
			||||||
 | 
					ZINC=
 | 
				
			||||||
 | 
					ZINCOUT=-I.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					OBJZ = adler32.o crc32.o deflate.o infback.o inffast.o inflate.o inftrees.o trees.o zutil.o
 | 
				
			||||||
 | 
					OBJG = compress.o uncompr.o gzclose.o gzlib.o gzread.o gzwrite.o
 | 
				
			||||||
 | 
					OBJC = $(OBJZ) $(OBJG)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					PIC_OBJZ = adler32.lo crc32.lo deflate.lo infback.lo inffast.lo inflate.lo inftrees.lo trees.lo zutil.lo
 | 
				
			||||||
 | 
					PIC_OBJG = compress.lo uncompr.lo gzclose.lo gzlib.lo gzread.lo gzwrite.lo
 | 
				
			||||||
 | 
					PIC_OBJC = $(PIC_OBJZ) $(PIC_OBJG)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# to use the asm code: make OBJA=match.o, PIC_OBJA=match.lo
 | 
				
			||||||
 | 
					OBJA =
 | 
				
			||||||
 | 
					PIC_OBJA =
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					OBJS = $(OBJC) $(OBJA)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					PIC_OBJS = $(PIC_OBJC) $(PIC_OBJA)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					all: static shared
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static: example$(EXE) minigzip$(EXE)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					shared: examplesh$(EXE) minigzipsh$(EXE)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					all64: example64$(EXE) minigzip64$(EXE)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					check: test
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					test: all teststatic testshared
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					teststatic: static
 | 
				
			||||||
 | 
						@TMPST=tmpst_$$; \
 | 
				
			||||||
 | 
						if echo hello world | ${QEMU_RUN} ./minigzip | ${QEMU_RUN} ./minigzip -d && ${QEMU_RUN} ./example $$TMPST ; then \
 | 
				
			||||||
 | 
						  echo '		*** zlib test OK ***'; \
 | 
				
			||||||
 | 
						else \
 | 
				
			||||||
 | 
						  echo '		*** zlib test FAILED ***'; false; \
 | 
				
			||||||
 | 
						fi
 | 
				
			||||||
 | 
						@rm -f tmpst_$$
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					testshared: shared
 | 
				
			||||||
 | 
						@LD_LIBRARY_PATH=`pwd`:$(LD_LIBRARY_PATH) ; export LD_LIBRARY_PATH; \
 | 
				
			||||||
 | 
						LD_LIBRARYN32_PATH=`pwd`:$(LD_LIBRARYN32_PATH) ; export LD_LIBRARYN32_PATH; \
 | 
				
			||||||
 | 
						DYLD_LIBRARY_PATH=`pwd`:$(DYLD_LIBRARY_PATH) ; export DYLD_LIBRARY_PATH; \
 | 
				
			||||||
 | 
						SHLIB_PATH=`pwd`:$(SHLIB_PATH) ; export SHLIB_PATH; \
 | 
				
			||||||
 | 
						TMPSH=tmpsh_$$; \
 | 
				
			||||||
 | 
						if echo hello world | ${QEMU_RUN} ./minigzipsh | ${QEMU_RUN} ./minigzipsh -d && ${QEMU_RUN} ./examplesh $$TMPSH; then \
 | 
				
			||||||
 | 
						  echo '		*** zlib shared test OK ***'; \
 | 
				
			||||||
 | 
						else \
 | 
				
			||||||
 | 
						  echo '		*** zlib shared test FAILED ***'; false; \
 | 
				
			||||||
 | 
						fi
 | 
				
			||||||
 | 
						@rm -f tmpsh_$$
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					test64: all64
 | 
				
			||||||
 | 
						@TMP64=tmp64_$$; \
 | 
				
			||||||
 | 
						if echo hello world | ${QEMU_RUN} ./minigzip64 | ${QEMU_RUN} ./minigzip64 -d && ${QEMU_RUN} ./example64 $$TMP64; then \
 | 
				
			||||||
 | 
						  echo '		*** zlib 64-bit test OK ***'; \
 | 
				
			||||||
 | 
						else \
 | 
				
			||||||
 | 
						  echo '		*** zlib 64-bit test FAILED ***'; false; \
 | 
				
			||||||
 | 
						fi
 | 
				
			||||||
 | 
						@rm -f tmp64_$$
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					infcover.o: $(SRCDIR)test/infcover.c $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINCOUT) -c -o $@ $(SRCDIR)test/infcover.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					infcover: infcover.o libz.a
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) -o $@ infcover.o libz.a
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					cover: infcover
 | 
				
			||||||
 | 
						rm -f *.gcda
 | 
				
			||||||
 | 
						${QEMU_RUN} ./infcover
 | 
				
			||||||
 | 
						gcov inf*.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					libz.a: $(OBJS)
 | 
				
			||||||
 | 
						$(AR) $(ARFLAGS) $@ $(OBJS)
 | 
				
			||||||
 | 
						-@ ($(RANLIB) $@ || true) >/dev/null 2>&1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					match.o: match.S
 | 
				
			||||||
 | 
						$(CPP) match.S > _match.s
 | 
				
			||||||
 | 
						$(CC) -c _match.s
 | 
				
			||||||
 | 
						mv _match.o match.o
 | 
				
			||||||
 | 
						rm -f _match.s
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					match.lo: match.S
 | 
				
			||||||
 | 
						$(CPP) match.S > _match.s
 | 
				
			||||||
 | 
						$(CC) -c -fPIC _match.s
 | 
				
			||||||
 | 
						mv _match.o match.lo
 | 
				
			||||||
 | 
						rm -f _match.s
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example.o: $(SRCDIR)test/example.c $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINCOUT) -c -o $@ $(SRCDIR)test/example.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip.o: $(SRCDIR)test/minigzip.c $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINCOUT) -c -o $@ $(SRCDIR)test/minigzip.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example64.o: $(SRCDIR)test/example.c $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINCOUT) -D_FILE_OFFSET_BITS=64 -c -o $@ $(SRCDIR)test/example.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip64.o: $(SRCDIR)test/minigzip.c $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINCOUT) -D_FILE_OFFSET_BITS=64 -c -o $@ $(SRCDIR)test/minigzip.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adler32.o: $(SRCDIR)adler32.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)adler32.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					crc32.o: $(SRCDIR)crc32.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)crc32.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					deflate.o: $(SRCDIR)deflate.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)deflate.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					infback.o: $(SRCDIR)infback.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)infback.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inffast.o: $(SRCDIR)inffast.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)inffast.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inflate.o: $(SRCDIR)inflate.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)inflate.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inftrees.o: $(SRCDIR)inftrees.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)inftrees.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					trees.o: $(SRCDIR)trees.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)trees.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zutil.o: $(SRCDIR)zutil.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)zutil.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					compress.o: $(SRCDIR)compress.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)compress.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					uncompr.o: $(SRCDIR)uncompr.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)uncompr.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzclose.o: $(SRCDIR)gzclose.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)gzclose.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzlib.o: $(SRCDIR)gzlib.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)gzlib.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzread.o: $(SRCDIR)gzread.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)gzread.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzwrite.o: $(SRCDIR)gzwrite.c
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(ZINC) -c -o $@ $(SRCDIR)gzwrite.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adler32.lo: $(SRCDIR)adler32.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/adler32.o $(SRCDIR)adler32.c
 | 
				
			||||||
 | 
						-@mv objs/adler32.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					crc32.lo: $(SRCDIR)crc32.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/crc32.o $(SRCDIR)crc32.c
 | 
				
			||||||
 | 
						-@mv objs/crc32.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					deflate.lo: $(SRCDIR)deflate.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/deflate.o $(SRCDIR)deflate.c
 | 
				
			||||||
 | 
						-@mv objs/deflate.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					infback.lo: $(SRCDIR)infback.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/infback.o $(SRCDIR)infback.c
 | 
				
			||||||
 | 
						-@mv objs/infback.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inffast.lo: $(SRCDIR)inffast.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/inffast.o $(SRCDIR)inffast.c
 | 
				
			||||||
 | 
						-@mv objs/inffast.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inflate.lo: $(SRCDIR)inflate.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/inflate.o $(SRCDIR)inflate.c
 | 
				
			||||||
 | 
						-@mv objs/inflate.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inftrees.lo: $(SRCDIR)inftrees.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/inftrees.o $(SRCDIR)inftrees.c
 | 
				
			||||||
 | 
						-@mv objs/inftrees.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					trees.lo: $(SRCDIR)trees.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/trees.o $(SRCDIR)trees.c
 | 
				
			||||||
 | 
						-@mv objs/trees.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zutil.lo: $(SRCDIR)zutil.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/zutil.o $(SRCDIR)zutil.c
 | 
				
			||||||
 | 
						-@mv objs/zutil.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					compress.lo: $(SRCDIR)compress.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/compress.o $(SRCDIR)compress.c
 | 
				
			||||||
 | 
						-@mv objs/compress.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					uncompr.lo: $(SRCDIR)uncompr.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/uncompr.o $(SRCDIR)uncompr.c
 | 
				
			||||||
 | 
						-@mv objs/uncompr.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzclose.lo: $(SRCDIR)gzclose.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/gzclose.o $(SRCDIR)gzclose.c
 | 
				
			||||||
 | 
						-@mv objs/gzclose.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzlib.lo: $(SRCDIR)gzlib.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/gzlib.o $(SRCDIR)gzlib.c
 | 
				
			||||||
 | 
						-@mv objs/gzlib.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzread.lo: $(SRCDIR)gzread.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/gzread.o $(SRCDIR)gzread.c
 | 
				
			||||||
 | 
						-@mv objs/gzread.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzwrite.lo: $(SRCDIR)gzwrite.c
 | 
				
			||||||
 | 
						-@mkdir objs 2>/dev/null || test -d objs
 | 
				
			||||||
 | 
						$(CC) $(SFLAGS) $(ZINC) -DPIC -c -o objs/gzwrite.o $(SRCDIR)gzwrite.c
 | 
				
			||||||
 | 
						-@mv objs/gzwrite.o $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					placebo $(SHAREDLIBV): $(PIC_OBJS) libz.a
 | 
				
			||||||
 | 
						$(LDSHARED) $(SFLAGS) -o $@ $(PIC_OBJS) $(LDSHAREDLIBC) $(LDFLAGS)
 | 
				
			||||||
 | 
						rm -f $(SHAREDLIB) $(SHAREDLIBM)
 | 
				
			||||||
 | 
						ln -s $@ $(SHAREDLIB)
 | 
				
			||||||
 | 
						ln -s $@ $(SHAREDLIBM)
 | 
				
			||||||
 | 
						-@rmdir objs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example$(EXE): example.o $(STATICLIB)
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(LDFLAGS) -o $@ example.o $(TEST_LIBS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip$(EXE): minigzip.o $(STATICLIB)
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(LDFLAGS) -o $@ minigzip.o $(TEST_LIBS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					examplesh$(EXE): example.o $(SHAREDLIBV)
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) -o $@ example.o $(LDFLAGS) -L. $(SHAREDLIBV)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzipsh$(EXE): minigzip.o $(SHAREDLIBV)
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) -o $@ minigzip.o $(LDFLAGS) -L. $(SHAREDLIBV)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example64$(EXE): example64.o $(STATICLIB)
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(LDFLAGS) -o $@ example64.o $(TEST_LIBS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip64$(EXE): minigzip64.o $(STATICLIB)
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) $(LDFLAGS) -o $@ minigzip64.o $(TEST_LIBS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					install-libs: $(LIBS)
 | 
				
			||||||
 | 
						-@if [ ! -d $(DESTDIR)$(exec_prefix)  ]; then mkdir -p $(DESTDIR)$(exec_prefix); fi
 | 
				
			||||||
 | 
						-@if [ ! -d $(DESTDIR)$(libdir)       ]; then mkdir -p $(DESTDIR)$(libdir); fi
 | 
				
			||||||
 | 
						-@if [ ! -d $(DESTDIR)$(sharedlibdir) ]; then mkdir -p $(DESTDIR)$(sharedlibdir); fi
 | 
				
			||||||
 | 
						-@if [ ! -d $(DESTDIR)$(man3dir)      ]; then mkdir -p $(DESTDIR)$(man3dir); fi
 | 
				
			||||||
 | 
						-@if [ ! -d $(DESTDIR)$(pkgconfigdir) ]; then mkdir -p $(DESTDIR)$(pkgconfigdir); fi
 | 
				
			||||||
 | 
						rm -f $(DESTDIR)$(libdir)/$(STATICLIB)
 | 
				
			||||||
 | 
						cp $(STATICLIB) $(DESTDIR)$(libdir)
 | 
				
			||||||
 | 
						chmod 644 $(DESTDIR)$(libdir)/$(STATICLIB)
 | 
				
			||||||
 | 
						-@($(RANLIB) $(DESTDIR)$(libdir)/libz.a || true) >/dev/null 2>&1
 | 
				
			||||||
 | 
						-@if test -n "$(SHAREDLIBV)"; then \
 | 
				
			||||||
 | 
						  rm -f $(DESTDIR)$(sharedlibdir)/$(SHAREDLIBV); \
 | 
				
			||||||
 | 
						  cp $(SHAREDLIBV) $(DESTDIR)$(sharedlibdir); \
 | 
				
			||||||
 | 
						  echo "cp $(SHAREDLIBV) $(DESTDIR)$(sharedlibdir)"; \
 | 
				
			||||||
 | 
						  chmod 755 $(DESTDIR)$(sharedlibdir)/$(SHAREDLIBV); \
 | 
				
			||||||
 | 
						  echo "chmod 755 $(DESTDIR)$(sharedlibdir)/$(SHAREDLIBV)"; \
 | 
				
			||||||
 | 
						  rm -f $(DESTDIR)$(sharedlibdir)/$(SHAREDLIB) $(DESTDIR)$(sharedlibdir)/$(SHAREDLIBM); \
 | 
				
			||||||
 | 
						  ln -s $(SHAREDLIBV) $(DESTDIR)$(sharedlibdir)/$(SHAREDLIB); \
 | 
				
			||||||
 | 
						  ln -s $(SHAREDLIBV) $(DESTDIR)$(sharedlibdir)/$(SHAREDLIBM); \
 | 
				
			||||||
 | 
						  ($(LDCONFIG) || true)  >/dev/null 2>&1; \
 | 
				
			||||||
 | 
						fi
 | 
				
			||||||
 | 
						rm -f $(DESTDIR)$(man3dir)/zlib.3
 | 
				
			||||||
 | 
						cp $(SRCDIR)zlib.3 $(DESTDIR)$(man3dir)
 | 
				
			||||||
 | 
						chmod 644 $(DESTDIR)$(man3dir)/zlib.3
 | 
				
			||||||
 | 
						rm -f $(DESTDIR)$(pkgconfigdir)/zlib.pc
 | 
				
			||||||
 | 
						cp zlib.pc $(DESTDIR)$(pkgconfigdir)
 | 
				
			||||||
 | 
						chmod 644 $(DESTDIR)$(pkgconfigdir)/zlib.pc
 | 
				
			||||||
 | 
					# The ranlib in install is needed on NeXTSTEP which checks file times
 | 
				
			||||||
 | 
					# ldconfig is for Linux
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					install: install-libs
 | 
				
			||||||
 | 
						-@if [ ! -d $(DESTDIR)$(includedir)   ]; then mkdir -p $(DESTDIR)$(includedir); fi
 | 
				
			||||||
 | 
						rm -f $(DESTDIR)$(includedir)/zlib.h $(DESTDIR)$(includedir)/zconf.h
 | 
				
			||||||
 | 
						cp $(SRCDIR)zlib.h zconf.h $(DESTDIR)$(includedir)
 | 
				
			||||||
 | 
						chmod 644 $(DESTDIR)$(includedir)/zlib.h $(DESTDIR)$(includedir)/zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					uninstall:
 | 
				
			||||||
 | 
						cd $(DESTDIR)$(includedir) && rm -f zlib.h zconf.h
 | 
				
			||||||
 | 
						cd $(DESTDIR)$(libdir) && rm -f libz.a; \
 | 
				
			||||||
 | 
						if test -n "$(SHAREDLIBV)" -a -f $(SHAREDLIBV); then \
 | 
				
			||||||
 | 
						  rm -f $(SHAREDLIBV) $(SHAREDLIB) $(SHAREDLIBM); \
 | 
				
			||||||
 | 
						fi
 | 
				
			||||||
 | 
						cd $(DESTDIR)$(man3dir) && rm -f zlib.3
 | 
				
			||||||
 | 
						cd $(DESTDIR)$(pkgconfigdir) && rm -f zlib.pc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					docs: zlib.3.pdf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zlib.3.pdf: $(SRCDIR)zlib.3
 | 
				
			||||||
 | 
						groff -mandoc -f H -T ps $(SRCDIR)zlib.3 | ps2pdf - $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zconf.h.cmakein: $(SRCDIR)zconf.h.in
 | 
				
			||||||
 | 
						-@ TEMPFILE=zconfh_$$; \
 | 
				
			||||||
 | 
						echo "/#define ZCONF_H/ a\\\\\n#cmakedefine Z_PREFIX\\\\\n#cmakedefine Z_HAVE_UNISTD_H\n" >> $$TEMPFILE &&\
 | 
				
			||||||
 | 
						sed -f $$TEMPFILE $(SRCDIR)zconf.h.in > $@ &&\
 | 
				
			||||||
 | 
						touch -r $(SRCDIR)zconf.h.in $@ &&\
 | 
				
			||||||
 | 
						rm $$TEMPFILE
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zconf: $(SRCDIR)zconf.h.in
 | 
				
			||||||
 | 
						cp -p $(SRCDIR)zconf.h.in zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minizip-test: static
 | 
				
			||||||
 | 
						cd contrib/minizip && { CC="$(CC)" CFLAGS="$(CFLAGS)" $(MAKE) test ; cd ../.. ; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minizip-clean:
 | 
				
			||||||
 | 
						cd contrib/minizip && { $(MAKE) clean ; cd ../.. ; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mostlyclean: clean
 | 
				
			||||||
 | 
					clean: minizip-clean
 | 
				
			||||||
 | 
						rm -f *.o *.lo *~ \
 | 
				
			||||||
 | 
						   example$(EXE) minigzip$(EXE) examplesh$(EXE) minigzipsh$(EXE) \
 | 
				
			||||||
 | 
						   example64$(EXE) minigzip64$(EXE) \
 | 
				
			||||||
 | 
						   infcover \
 | 
				
			||||||
 | 
						   libz.* foo.gz so_locations \
 | 
				
			||||||
 | 
						   _match.s maketree contrib/infback9/*.o
 | 
				
			||||||
 | 
						rm -rf objs
 | 
				
			||||||
 | 
						rm -f *.gcda *.gcno *.gcov
 | 
				
			||||||
 | 
						rm -f contrib/infback9/*.gcda contrib/infback9/*.gcno contrib/infback9/*.gcov
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					maintainer-clean: distclean
 | 
				
			||||||
 | 
					distclean: clean zconf zconf.h.cmakein
 | 
				
			||||||
 | 
						rm -f Makefile zlib.pc configure.log
 | 
				
			||||||
 | 
						-@rm -f .DS_Store
 | 
				
			||||||
 | 
						@if [ -f Makefile.in ]; then \
 | 
				
			||||||
 | 
						printf 'all:\n\t-@echo "Please use ./configure first.  Thank you."\n' > Makefile ; \
 | 
				
			||||||
 | 
						printf '\ndistclean:\n\tmake -f Makefile.in distclean\n' >> Makefile ; \
 | 
				
			||||||
 | 
						touch -r $(SRCDIR)Makefile.in Makefile ; fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					tags:
 | 
				
			||||||
 | 
						etags $(SRCDIR)*.[ch]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adler32.o zutil.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
					gzclose.o gzlib.o gzread.o gzwrite.o: $(SRCDIR)zlib.h zconf.h $(SRCDIR)gzguts.h
 | 
				
			||||||
 | 
					compress.o example.o minigzip.o uncompr.o: $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
					crc32.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)crc32.h
 | 
				
			||||||
 | 
					deflate.o: $(SRCDIR)deflate.h $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
					infback.o inflate.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h $(SRCDIR)inflate.h $(SRCDIR)inffast.h $(SRCDIR)inffixed.h
 | 
				
			||||||
 | 
					inffast.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h $(SRCDIR)inflate.h $(SRCDIR)inffast.h
 | 
				
			||||||
 | 
					inftrees.o: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h
 | 
				
			||||||
 | 
					trees.o: $(SRCDIR)deflate.h $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)trees.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adler32.lo zutil.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
					gzclose.lo gzlib.lo gzread.lo gzwrite.lo: $(SRCDIR)zlib.h zconf.h $(SRCDIR)gzguts.h
 | 
				
			||||||
 | 
					compress.lo example.lo minigzip.lo uncompr.lo: $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
					crc32.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)crc32.h
 | 
				
			||||||
 | 
					deflate.lo: $(SRCDIR)deflate.h $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h
 | 
				
			||||||
 | 
					infback.lo inflate.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h $(SRCDIR)inflate.h $(SRCDIR)inffast.h $(SRCDIR)inffixed.h
 | 
				
			||||||
 | 
					inffast.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h $(SRCDIR)inflate.h $(SRCDIR)inffast.h
 | 
				
			||||||
 | 
					inftrees.lo: $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)inftrees.h
 | 
				
			||||||
 | 
					trees.lo: $(SRCDIR)deflate.h $(SRCDIR)zutil.h $(SRCDIR)zlib.h zconf.h $(SRCDIR)trees.h
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,117 @@
 | 
				
			||||||
 | 
					ZLIB DATA COMPRESSION LIBRARY
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zlib 1.3.1 is a general purpose data compression library.  All the code is
 | 
				
			||||||
 | 
					thread safe.  The data format used by the zlib library is described by RFCs
 | 
				
			||||||
 | 
					(Request for Comments) 1950 to 1952 in the files
 | 
				
			||||||
 | 
					http://tools.ietf.org/html/rfc1950 (zlib format), rfc1951 (deflate format) and
 | 
				
			||||||
 | 
					rfc1952 (gzip format).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					All functions of the compression library are documented in the file zlib.h
 | 
				
			||||||
 | 
					(volunteer to write man pages welcome, contact zlib@gzip.org).  A usage example
 | 
				
			||||||
 | 
					of the library is given in the file test/example.c which also tests that
 | 
				
			||||||
 | 
					the library is working correctly.  Another example is given in the file
 | 
				
			||||||
 | 
					test/minigzip.c.  The compression library itself is composed of all source
 | 
				
			||||||
 | 
					files in the root directory.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					To compile all files and run the test program, follow the instructions given at
 | 
				
			||||||
 | 
					the top of Makefile.in.  In short "./configure; make test", and if that goes
 | 
				
			||||||
 | 
					well, "make install" should work for most flavors of Unix.  For Windows, use
 | 
				
			||||||
 | 
					one of the special makefiles in win32/ or contrib/vstudio/ .  For VMS, use
 | 
				
			||||||
 | 
					make_vms.com.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Questions about zlib should be sent to <zlib@gzip.org>, or to Gilles Vollant
 | 
				
			||||||
 | 
					<info@winimage.com> for the Windows DLL version.  The zlib home page is
 | 
				
			||||||
 | 
					http://zlib.net/ .  Before reporting a problem, please check this site to
 | 
				
			||||||
 | 
					verify that you have the latest version of zlib; otherwise get the latest
 | 
				
			||||||
 | 
					version and check whether the problem still exists or not.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					PLEASE read the zlib FAQ http://zlib.net/zlib_faq.html before asking for help.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Mark Nelson <markn@ieee.org> wrote an article about zlib for the Jan.  1997
 | 
				
			||||||
 | 
					issue of Dr.  Dobb's Journal; a copy of the article is available at
 | 
				
			||||||
 | 
					https://marknelson.us/posts/1997/01/01/zlib-engine.html .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The changes made in version 1.3.1 are documented in the file ChangeLog.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Unsupported third party contributions are provided in directory contrib/ .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zlib is available in Java using the java.util.zip package. Follow the API
 | 
				
			||||||
 | 
					Documentation link at: https://docs.oracle.com/search/?q=java.util.zip .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					A Perl interface to zlib and bzip2 written by Paul Marquess <pmqs@cpan.org>
 | 
				
			||||||
 | 
					can be found at https://github.com/pmqs/IO-Compress .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					A Python interface to zlib written by A.M. Kuchling <amk@amk.ca> is
 | 
				
			||||||
 | 
					available in Python 1.5 and later versions, see
 | 
				
			||||||
 | 
					http://docs.python.org/library/zlib.html .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zlib is built into tcl: http://wiki.tcl.tk/4610 .
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					An experimental package to read and write files in .zip format, written on top
 | 
				
			||||||
 | 
					of zlib by Gilles Vollant <info@winimage.com>, is available in the
 | 
				
			||||||
 | 
					contrib/minizip directory of zlib.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Notes for some targets:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- For Windows DLL versions, please see win32/DLL_FAQ.txt
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- For 64-bit Irix, deflate.c must be compiled without any optimization. With
 | 
				
			||||||
 | 
					  -O, one libpng test fails. The test works in 32 bit mode (with the -n32
 | 
				
			||||||
 | 
					  compiler flag). The compiler bug has been reported to SGI.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- zlib doesn't work with gcc 2.6.3 on a DEC 3000/300LX under OSF/1 2.1 it works
 | 
				
			||||||
 | 
					  when compiled with cc.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- On Digital Unix 4.0D (formerly OSF/1) on AlphaServer, the cc option -std1 is
 | 
				
			||||||
 | 
					  necessary to get gzprintf working correctly. This is done by configure.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- zlib doesn't work on HP-UX 9.05 with some versions of /bin/cc. It works with
 | 
				
			||||||
 | 
					  other compilers. Use "make test" to check your compiler.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- gzdopen is not supported on RISCOS or BEOS.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- For PalmOs, see http://palmzlib.sourceforge.net/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Acknowledgments:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  The deflate format used by zlib was defined by Phil Katz.  The deflate and
 | 
				
			||||||
 | 
					  zlib specifications were written by L.  Peter Deutsch.  Thanks to all the
 | 
				
			||||||
 | 
					  people who reported problems and suggested various improvements in zlib; they
 | 
				
			||||||
 | 
					  are too numerous to cite here.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Copyright notice:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 (C) 1995-2024 Jean-loup Gailly and Mark Adler
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  This software is provided 'as-is', without any express or implied
 | 
				
			||||||
 | 
					  warranty.  In no event will the authors be held liable for any damages
 | 
				
			||||||
 | 
					  arising from the use of this software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Permission is granted to anyone to use this software for any purpose,
 | 
				
			||||||
 | 
					  including commercial applications, and to alter it and redistribute it
 | 
				
			||||||
 | 
					  freely, subject to the following restrictions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  1. The origin of this software must not be misrepresented; you must not
 | 
				
			||||||
 | 
					     claim that you wrote the original software. If you use this software
 | 
				
			||||||
 | 
					     in a product, an acknowledgment in the product documentation would be
 | 
				
			||||||
 | 
					     appreciated but is not required.
 | 
				
			||||||
 | 
					  2. Altered source versions must be plainly marked as such, and must not be
 | 
				
			||||||
 | 
					     misrepresented as being the original software.
 | 
				
			||||||
 | 
					  3. This notice may not be removed or altered from any source distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Jean-loup Gailly        Mark Adler
 | 
				
			||||||
 | 
					  jloup@gzip.org          madler@alumni.caltech.edu
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					If you use the zlib library in a product, we would appreciate *not* receiving
 | 
				
			||||||
 | 
					lengthy legal documents to sign.  The sources are provided for free but without
 | 
				
			||||||
 | 
					warranty of any kind.  The library has been entirely written by Jean-loup
 | 
				
			||||||
 | 
					Gailly and Mark Adler; it does not include third-party code.  We make all
 | 
				
			||||||
 | 
					contributions to and distributions of this project solely in our personal
 | 
				
			||||||
 | 
					capacity, and are not conveying any rights to any intellectual property of
 | 
				
			||||||
 | 
					any third parties.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					If you redistribute modified sources, we would appreciate that you include in
 | 
				
			||||||
 | 
					the file ChangeLog history information documenting your changes.  Please read
 | 
				
			||||||
 | 
					the FAQ for more information on the distribution of modified source versions.
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,164 @@
 | 
				
			||||||
 | 
					/* adler32.c -- compute the Adler-32 checksum of a data stream
 | 
				
			||||||
 | 
					 * Copyright (C) 1995-2011, 2016 Mark Adler
 | 
				
			||||||
 | 
					 * For conditions of distribution and use, see copyright notice in zlib.h
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* @(#) $Id$ */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "zutil.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define BASE 65521U     /* largest prime smaller than 65536 */
 | 
				
			||||||
 | 
					#define NMAX 5552
 | 
				
			||||||
 | 
					/* NMAX is the largest n such that 255n(n+1)/2 + (n+1)(BASE-1) <= 2^32-1 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define DO1(buf,i)  {adler += (buf)[i]; sum2 += adler;}
 | 
				
			||||||
 | 
					#define DO2(buf,i)  DO1(buf,i); DO1(buf,i+1);
 | 
				
			||||||
 | 
					#define DO4(buf,i)  DO2(buf,i); DO2(buf,i+2);
 | 
				
			||||||
 | 
					#define DO8(buf,i)  DO4(buf,i); DO4(buf,i+4);
 | 
				
			||||||
 | 
					#define DO16(buf)   DO8(buf,0); DO8(buf,8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* use NO_DIVIDE if your processor does not do division in hardware --
 | 
				
			||||||
 | 
					   try it both ways to see which is faster */
 | 
				
			||||||
 | 
					#ifdef NO_DIVIDE
 | 
				
			||||||
 | 
					/* note that this assumes BASE is 65521, where 65536 % 65521 == 15
 | 
				
			||||||
 | 
					   (thank you to John Reiser for pointing this out) */
 | 
				
			||||||
 | 
					#  define CHOP(a) \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        unsigned long tmp = a >> 16; \
 | 
				
			||||||
 | 
					        a &= 0xffffUL; \
 | 
				
			||||||
 | 
					        a += (tmp << 4) - tmp; \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					#  define MOD28(a) \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        CHOP(a); \
 | 
				
			||||||
 | 
					        if (a >= BASE) a -= BASE; \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					#  define MOD(a) \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        CHOP(a); \
 | 
				
			||||||
 | 
					        MOD28(a); \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					#  define MOD63(a) \
 | 
				
			||||||
 | 
					    do { /* this assumes a is not negative */ \
 | 
				
			||||||
 | 
					        z_off64_t tmp = a >> 32; \
 | 
				
			||||||
 | 
					        a &= 0xffffffffL; \
 | 
				
			||||||
 | 
					        a += (tmp << 8) - (tmp << 5) + tmp; \
 | 
				
			||||||
 | 
					        tmp = a >> 16; \
 | 
				
			||||||
 | 
					        a &= 0xffffL; \
 | 
				
			||||||
 | 
					        a += (tmp << 4) - tmp; \
 | 
				
			||||||
 | 
					        tmp = a >> 16; \
 | 
				
			||||||
 | 
					        a &= 0xffffL; \
 | 
				
			||||||
 | 
					        a += (tmp << 4) - tmp; \
 | 
				
			||||||
 | 
					        if (a >= BASE) a -= BASE; \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					#else
 | 
				
			||||||
 | 
					#  define MOD(a) a %= BASE
 | 
				
			||||||
 | 
					#  define MOD28(a) a %= BASE
 | 
				
			||||||
 | 
					#  define MOD63(a) a %= BASE
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* ========================================================================= */
 | 
				
			||||||
 | 
					uLong ZEXPORT adler32_z(uLong adler, const Bytef *buf, z_size_t len) {
 | 
				
			||||||
 | 
					    unsigned long sum2;
 | 
				
			||||||
 | 
					    unsigned n;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* split Adler-32 into component sums */
 | 
				
			||||||
 | 
					    sum2 = (adler >> 16) & 0xffff;
 | 
				
			||||||
 | 
					    adler &= 0xffff;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* in case user likes doing a byte at a time, keep it fast */
 | 
				
			||||||
 | 
					    if (len == 1) {
 | 
				
			||||||
 | 
					        adler += buf[0];
 | 
				
			||||||
 | 
					        if (adler >= BASE)
 | 
				
			||||||
 | 
					            adler -= BASE;
 | 
				
			||||||
 | 
					        sum2 += adler;
 | 
				
			||||||
 | 
					        if (sum2 >= BASE)
 | 
				
			||||||
 | 
					            sum2 -= BASE;
 | 
				
			||||||
 | 
					        return adler | (sum2 << 16);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* initial Adler-32 value (deferred check for len == 1 speed) */
 | 
				
			||||||
 | 
					    if (buf == Z_NULL)
 | 
				
			||||||
 | 
					        return 1L;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* in case short lengths are provided, keep it somewhat fast */
 | 
				
			||||||
 | 
					    if (len < 16) {
 | 
				
			||||||
 | 
					        while (len--) {
 | 
				
			||||||
 | 
					            adler += *buf++;
 | 
				
			||||||
 | 
					            sum2 += adler;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (adler >= BASE)
 | 
				
			||||||
 | 
					            adler -= BASE;
 | 
				
			||||||
 | 
					        MOD28(sum2);            /* only added so many BASE's */
 | 
				
			||||||
 | 
					        return adler | (sum2 << 16);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* do length NMAX blocks -- requires just one modulo operation */
 | 
				
			||||||
 | 
					    while (len >= NMAX) {
 | 
				
			||||||
 | 
					        len -= NMAX;
 | 
				
			||||||
 | 
					        n = NMAX / 16;          /* NMAX is divisible by 16 */
 | 
				
			||||||
 | 
					        do {
 | 
				
			||||||
 | 
					            DO16(buf);          /* 16 sums unrolled */
 | 
				
			||||||
 | 
					            buf += 16;
 | 
				
			||||||
 | 
					        } while (--n);
 | 
				
			||||||
 | 
					        MOD(adler);
 | 
				
			||||||
 | 
					        MOD(sum2);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* do remaining bytes (less than NMAX, still just one modulo) */
 | 
				
			||||||
 | 
					    if (len) {                  /* avoid modulos if none remaining */
 | 
				
			||||||
 | 
					        while (len >= 16) {
 | 
				
			||||||
 | 
					            len -= 16;
 | 
				
			||||||
 | 
					            DO16(buf);
 | 
				
			||||||
 | 
					            buf += 16;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        while (len--) {
 | 
				
			||||||
 | 
					            adler += *buf++;
 | 
				
			||||||
 | 
					            sum2 += adler;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        MOD(adler);
 | 
				
			||||||
 | 
					        MOD(sum2);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* return recombined sums */
 | 
				
			||||||
 | 
					    return adler | (sum2 << 16);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* ========================================================================= */
 | 
				
			||||||
 | 
					uLong ZEXPORT adler32(uLong adler, const Bytef *buf, uInt len) {
 | 
				
			||||||
 | 
					    return adler32_z(adler, buf, len);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* ========================================================================= */
 | 
				
			||||||
 | 
					local uLong adler32_combine_(uLong adler1, uLong adler2, z_off64_t len2) {
 | 
				
			||||||
 | 
					    unsigned long sum1;
 | 
				
			||||||
 | 
					    unsigned long sum2;
 | 
				
			||||||
 | 
					    unsigned rem;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* for negative len, return invalid adler32 as a clue for debugging */
 | 
				
			||||||
 | 
					    if (len2 < 0)
 | 
				
			||||||
 | 
					        return 0xffffffffUL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* the derivation of this formula is left as an exercise for the reader */
 | 
				
			||||||
 | 
					    MOD63(len2);                /* assumes len2 >= 0 */
 | 
				
			||||||
 | 
					    rem = (unsigned)len2;
 | 
				
			||||||
 | 
					    sum1 = adler1 & 0xffff;
 | 
				
			||||||
 | 
					    sum2 = rem * sum1;
 | 
				
			||||||
 | 
					    MOD(sum2);
 | 
				
			||||||
 | 
					    sum1 += (adler2 & 0xffff) + BASE - 1;
 | 
				
			||||||
 | 
					    sum2 += ((adler1 >> 16) & 0xffff) + ((adler2 >> 16) & 0xffff) + BASE - rem;
 | 
				
			||||||
 | 
					    if (sum1 >= BASE) sum1 -= BASE;
 | 
				
			||||||
 | 
					    if (sum1 >= BASE) sum1 -= BASE;
 | 
				
			||||||
 | 
					    if (sum2 >= ((unsigned long)BASE << 1)) sum2 -= ((unsigned long)BASE << 1);
 | 
				
			||||||
 | 
					    if (sum2 >= BASE) sum2 -= BASE;
 | 
				
			||||||
 | 
					    return sum1 | (sum2 << 16);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* ========================================================================= */
 | 
				
			||||||
 | 
					uLong ZEXPORT adler32_combine(uLong adler1, uLong adler2, z_off_t len2) {
 | 
				
			||||||
 | 
					    return adler32_combine_(adler1, adler2, len2);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					uLong ZEXPORT adler32_combine64(uLong adler1, uLong adler2, z_off64_t len2) {
 | 
				
			||||||
 | 
					    return adler32_combine_(adler1, adler2, len2);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,69 @@
 | 
				
			||||||
 | 
					# Amiga powerUP (TM) Makefile
 | 
				
			||||||
 | 
					# makefile for libpng and SAS C V6.58/7.00 PPC compiler
 | 
				
			||||||
 | 
					# Copyright (C) 1998 by Andreas R. Kleinert
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LIBNAME	= libzip.a
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CC	= scppc
 | 
				
			||||||
 | 
					CFLAGS	= NOSTKCHK NOSINT OPTIMIZE OPTGO OPTPEEP OPTINLOCAL OPTINL \
 | 
				
			||||||
 | 
						  OPTLOOP OPTRDEP=8 OPTDEP=8 OPTCOMP=8 NOVER
 | 
				
			||||||
 | 
					AR	= ppc-amigaos-ar cr
 | 
				
			||||||
 | 
					RANLIB	= ppc-amigaos-ranlib
 | 
				
			||||||
 | 
					LD	= ppc-amigaos-ld -r
 | 
				
			||||||
 | 
					LDFLAGS	= -o
 | 
				
			||||||
 | 
					LDLIBS	= LIB:scppc.a LIB:end.o
 | 
				
			||||||
 | 
					RM	= delete quiet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					OBJS = adler32.o compress.o crc32.o gzclose.o gzlib.o gzread.o gzwrite.o \
 | 
				
			||||||
 | 
					       uncompr.o deflate.o trees.o zutil.o inflate.o infback.o inftrees.o inffast.o
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					TEST_OBJS = example.o minigzip.o
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					all: example minigzip
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					check: test
 | 
				
			||||||
 | 
					test: all
 | 
				
			||||||
 | 
						example
 | 
				
			||||||
 | 
						echo hello world | minigzip | minigzip -d
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					$(LIBNAME): $(OBJS)
 | 
				
			||||||
 | 
						$(AR) $@ $(OBJS)
 | 
				
			||||||
 | 
						-$(RANLIB) $@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example: example.o $(LIBNAME)
 | 
				
			||||||
 | 
						$(LD) $(LDFLAGS) $@ LIB:c_ppc.o $@.o $(LIBNAME) $(LDLIBS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip: minigzip.o $(LIBNAME)
 | 
				
			||||||
 | 
						$(LD) $(LDFLAGS) $@ LIB:c_ppc.o $@.o $(LIBNAME) $(LDLIBS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mostlyclean: clean
 | 
				
			||||||
 | 
					clean:
 | 
				
			||||||
 | 
						$(RM) *.o example minigzip $(LIBNAME) foo.gz
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zip:
 | 
				
			||||||
 | 
						zip -ul9 zlib README ChangeLog Makefile Make????.??? Makefile.?? \
 | 
				
			||||||
 | 
						  descrip.mms *.[ch]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					tgz:
 | 
				
			||||||
 | 
						cd ..; tar cfz zlib/zlib.tgz zlib/README zlib/ChangeLog zlib/Makefile \
 | 
				
			||||||
 | 
						  zlib/Make????.??? zlib/Makefile.?? zlib/descrip.mms zlib/*.[ch]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# DO NOT DELETE THIS LINE -- make depend depends on it.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adler32.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					compress.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					crc32.o: crc32.h zlib.h zconf.h
 | 
				
			||||||
 | 
					deflate.o: deflate.h zutil.h zlib.h zconf.h
 | 
				
			||||||
 | 
					example.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					gzclose.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					gzlib.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					gzread.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					gzwrite.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					inffast.o: zutil.h zlib.h zconf.h inftrees.h inflate.h inffast.h
 | 
				
			||||||
 | 
					inflate.o: zutil.h zlib.h zconf.h inftrees.h inflate.h inffast.h
 | 
				
			||||||
 | 
					infback.o: zutil.h zlib.h zconf.h inftrees.h inflate.h inffast.h
 | 
				
			||||||
 | 
					inftrees.o: zutil.h zlib.h zconf.h inftrees.h
 | 
				
			||||||
 | 
					minigzip.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					trees.o: deflate.h zutil.h zlib.h zconf.h trees.h
 | 
				
			||||||
 | 
					uncompr.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					zutil.o: zutil.h zlib.h zconf.h
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,68 @@
 | 
				
			||||||
 | 
					# SMakefile for zlib
 | 
				
			||||||
 | 
					# Modified from the standard UNIX Makefile Copyright Jean-loup Gailly
 | 
				
			||||||
 | 
					# Osma Ahvenlampi <Osma.Ahvenlampi@hut.fi>
 | 
				
			||||||
 | 
					# Amiga, SAS/C 6.56 & Smake
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CC=sc
 | 
				
			||||||
 | 
					CFLAGS=OPT
 | 
				
			||||||
 | 
					#CFLAGS=OPT CPU=68030
 | 
				
			||||||
 | 
					#CFLAGS=DEBUG=LINE
 | 
				
			||||||
 | 
					LDFLAGS=LIB z.lib
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					SCOPTIONS=OPTSCHED OPTINLINE OPTALIAS OPTTIME OPTINLOCAL STRMERGE \
 | 
				
			||||||
 | 
					       NOICONS PARMS=BOTH NOSTACKCHECK UTILLIB NOVERSION ERRORREXX \
 | 
				
			||||||
 | 
					       DEF=POSTINC
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					OBJS = adler32.o compress.o crc32.o gzclose.o gzlib.o gzread.o gzwrite.o \
 | 
				
			||||||
 | 
					       uncompr.o deflate.o trees.o zutil.o inflate.o infback.o inftrees.o inffast.o
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					TEST_OBJS = example.o minigzip.o
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					all: SCOPTIONS example minigzip
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					check: test
 | 
				
			||||||
 | 
					test: all
 | 
				
			||||||
 | 
						example
 | 
				
			||||||
 | 
						echo hello world | minigzip | minigzip -d
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					install: z.lib
 | 
				
			||||||
 | 
						copy clone zlib.h zconf.h INCLUDE:
 | 
				
			||||||
 | 
						copy clone z.lib LIB:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					z.lib: $(OBJS)
 | 
				
			||||||
 | 
						oml z.lib r $(OBJS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example: example.o z.lib
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) LINK TO $@ example.o $(LDFLAGS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip: minigzip.o z.lib
 | 
				
			||||||
 | 
						$(CC) $(CFLAGS) LINK TO $@ minigzip.o $(LDFLAGS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mostlyclean: clean
 | 
				
			||||||
 | 
					clean:
 | 
				
			||||||
 | 
						-delete force quiet example minigzip *.o z.lib foo.gz *.lnk SCOPTIONS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					SCOPTIONS: Makefile.sas
 | 
				
			||||||
 | 
						copy to $@ <from <
 | 
				
			||||||
 | 
					$(SCOPTIONS)
 | 
				
			||||||
 | 
					<
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# DO NOT DELETE THIS LINE -- make depend depends on it.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adler32.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					compress.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					crc32.o: crc32.h zlib.h zconf.h
 | 
				
			||||||
 | 
					deflate.o: deflate.h zutil.h zlib.h zconf.h
 | 
				
			||||||
 | 
					example.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					gzclose.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					gzlib.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					gzread.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					gzwrite.o: zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					inffast.o: zutil.h zlib.h zconf.h inftrees.h inflate.h inffast.h
 | 
				
			||||||
 | 
					inflate.o: zutil.h zlib.h zconf.h inftrees.h inflate.h inffast.h
 | 
				
			||||||
 | 
					infback.o: zutil.h zlib.h zconf.h inftrees.h inflate.h inffast.h
 | 
				
			||||||
 | 
					inftrees.o: zutil.h zlib.h zconf.h inftrees.h
 | 
				
			||||||
 | 
					minigzip.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					trees.o: deflate.h zutil.h zlib.h zconf.h trees.h
 | 
				
			||||||
 | 
					uncompr.o: zlib.h zconf.h
 | 
				
			||||||
 | 
					zutil.o: zutil.h zlib.h zconf.h
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,75 @@
 | 
				
			||||||
 | 
					/* compress.c -- compress a memory buffer
 | 
				
			||||||
 | 
					 * Copyright (C) 1995-2005, 2014, 2016 Jean-loup Gailly, Mark Adler
 | 
				
			||||||
 | 
					 * For conditions of distribution and use, see copyright notice in zlib.h
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* @(#) $Id$ */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define ZLIB_INTERNAL
 | 
				
			||||||
 | 
					#include "zlib.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* ===========================================================================
 | 
				
			||||||
 | 
					     Compresses the source buffer into the destination buffer. The level
 | 
				
			||||||
 | 
					   parameter has the same meaning as in deflateInit.  sourceLen is the byte
 | 
				
			||||||
 | 
					   length of the source buffer. Upon entry, destLen is the total size of the
 | 
				
			||||||
 | 
					   destination buffer, which must be at least 0.1% larger than sourceLen plus
 | 
				
			||||||
 | 
					   12 bytes. Upon exit, destLen is the actual size of the compressed buffer.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					     compress2 returns Z_OK if success, Z_MEM_ERROR if there was not enough
 | 
				
			||||||
 | 
					   memory, Z_BUF_ERROR if there was not enough room in the output buffer,
 | 
				
			||||||
 | 
					   Z_STREAM_ERROR if the level parameter is invalid.
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					int ZEXPORT compress2(Bytef *dest, uLongf *destLen, const Bytef *source,
 | 
				
			||||||
 | 
					                      uLong sourceLen, int level) {
 | 
				
			||||||
 | 
					    z_stream stream;
 | 
				
			||||||
 | 
					    int err;
 | 
				
			||||||
 | 
					    const uInt max = (uInt)-1;
 | 
				
			||||||
 | 
					    uLong left;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    left = *destLen;
 | 
				
			||||||
 | 
					    *destLen = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    stream.zalloc = (alloc_func)0;
 | 
				
			||||||
 | 
					    stream.zfree = (free_func)0;
 | 
				
			||||||
 | 
					    stream.opaque = (voidpf)0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    err = deflateInit(&stream, level);
 | 
				
			||||||
 | 
					    if (err != Z_OK) return err;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    stream.next_out = dest;
 | 
				
			||||||
 | 
					    stream.avail_out = 0;
 | 
				
			||||||
 | 
					    stream.next_in = (z_const Bytef *)source;
 | 
				
			||||||
 | 
					    stream.avail_in = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    do {
 | 
				
			||||||
 | 
					        if (stream.avail_out == 0) {
 | 
				
			||||||
 | 
					            stream.avail_out = left > (uLong)max ? max : (uInt)left;
 | 
				
			||||||
 | 
					            left -= stream.avail_out;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (stream.avail_in == 0) {
 | 
				
			||||||
 | 
					            stream.avail_in = sourceLen > (uLong)max ? max : (uInt)sourceLen;
 | 
				
			||||||
 | 
					            sourceLen -= stream.avail_in;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        err = deflate(&stream, sourceLen ? Z_NO_FLUSH : Z_FINISH);
 | 
				
			||||||
 | 
					    } while (err == Z_OK);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    *destLen = stream.total_out;
 | 
				
			||||||
 | 
					    deflateEnd(&stream);
 | 
				
			||||||
 | 
					    return err == Z_STREAM_END ? Z_OK : err;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* ===========================================================================
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					int ZEXPORT compress(Bytef *dest, uLongf *destLen, const Bytef *source,
 | 
				
			||||||
 | 
					                     uLong sourceLen) {
 | 
				
			||||||
 | 
					    return compress2(dest, destLen, source, sourceLen, Z_DEFAULT_COMPRESSION);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* ===========================================================================
 | 
				
			||||||
 | 
					     If the default memLevel or windowBits for deflateInit() is changed, then
 | 
				
			||||||
 | 
					   this function needs to be updated.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					uLong ZEXPORT compressBound(uLong sourceLen) {
 | 
				
			||||||
 | 
					    return sourceLen + (sourceLen >> 12) + (sourceLen >> 14) +
 | 
				
			||||||
 | 
					           (sourceLen >> 25) + 13;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,929 @@
 | 
				
			||||||
 | 
					#!/bin/sh
 | 
				
			||||||
 | 
					# configure script for zlib.
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# Normally configure builds both a static and a shared library.
 | 
				
			||||||
 | 
					# If you want to build just a static library, use: ./configure --static
 | 
				
			||||||
 | 
					#
 | 
				
			||||||
 | 
					# To impose specific compiler or flags or install directory, use for example:
 | 
				
			||||||
 | 
					#    prefix=$HOME CC=cc CFLAGS="-O4" ./configure
 | 
				
			||||||
 | 
					# or for csh/tcsh users:
 | 
				
			||||||
 | 
					#    (setenv prefix $HOME; setenv CC cc; setenv CFLAGS "-O4"; ./configure)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Incorrect settings of CC or CFLAGS may prevent creating a shared library.
 | 
				
			||||||
 | 
					# If you have problems, try without defining CC and CFLAGS before reporting
 | 
				
			||||||
 | 
					# an error.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# start off configure.log
 | 
				
			||||||
 | 
					echo -------------------- >> configure.log
 | 
				
			||||||
 | 
					echo $0 $* >> configure.log
 | 
				
			||||||
 | 
					date >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# get source directory
 | 
				
			||||||
 | 
					SRCDIR=`dirname $0`
 | 
				
			||||||
 | 
					if test $SRCDIR = "."; then
 | 
				
			||||||
 | 
					    ZINC=""
 | 
				
			||||||
 | 
					    ZINCOUT="-I."
 | 
				
			||||||
 | 
					    SRCDIR=""
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					    ZINC='-I. -include zconf.h'
 | 
				
			||||||
 | 
					    ZINCOUT='-I. -I$(SRCDIR)'
 | 
				
			||||||
 | 
					    SRCDIR="$SRCDIR/"
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# set command prefix for cross-compilation
 | 
				
			||||||
 | 
					if [ -n "${CHOST}" ]; then
 | 
				
			||||||
 | 
					    uname=${CHOST}
 | 
				
			||||||
 | 
					    mname=${CHOST}
 | 
				
			||||||
 | 
					    CROSS_PREFIX="${CHOST}-"
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					    mname=`(uname -a || echo unknown) 2>/dev/null`
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# destination name for static library
 | 
				
			||||||
 | 
					STATICLIB=libz.a
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# extract zlib version numbers from zlib.h
 | 
				
			||||||
 | 
					VER=`sed -n -e '/VERSION "/s/.*"\(.*\)".*/\1/p' < ${SRCDIR}zlib.h`
 | 
				
			||||||
 | 
					VER3=`echo ${VER}|sed -n -e 's/\([0-9]\{1,\}\(\\.[0-9]\{1,\}\)\{1,2\}\).*/\1/p'`
 | 
				
			||||||
 | 
					VER1=`echo ${VER}|sed -n -e 's/\([0-9]\{1,\}\)\\..*/\1/p'`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# establish commands for library building
 | 
				
			||||||
 | 
					if "${CROSS_PREFIX}ar" --version >/dev/null 2>/dev/null || test $? -lt 126; then
 | 
				
			||||||
 | 
					    AR=${AR-"${CROSS_PREFIX}ar"}
 | 
				
			||||||
 | 
					    test -n "${CROSS_PREFIX}" && echo Using ${AR} | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					    AR=${AR-"ar"}
 | 
				
			||||||
 | 
					    test -n "${CROSS_PREFIX}" && echo Using ${AR} | tee -a configure.log
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					ARFLAGS=${ARFLAGS-"rc"}
 | 
				
			||||||
 | 
					if "${CROSS_PREFIX}ranlib" --version >/dev/null 2>/dev/null || test $? -lt 126; then
 | 
				
			||||||
 | 
					    RANLIB=${RANLIB-"${CROSS_PREFIX}ranlib"}
 | 
				
			||||||
 | 
					    test -n "${CROSS_PREFIX}" && echo Using ${RANLIB} | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					    RANLIB=${RANLIB-"ranlib"}
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					if "${CROSS_PREFIX}nm" --version >/dev/null 2>/dev/null || test $? -lt 126; then
 | 
				
			||||||
 | 
					    NM=${NM-"${CROSS_PREFIX}nm"}
 | 
				
			||||||
 | 
					    test -n "${CROSS_PREFIX}" && echo Using ${NM} | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					    NM=${NM-"nm"}
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# set defaults before processing command line options
 | 
				
			||||||
 | 
					LDCONFIG=${LDCONFIG-"ldconfig"}
 | 
				
			||||||
 | 
					LDSHAREDLIBC="${LDSHAREDLIBC--lc}"
 | 
				
			||||||
 | 
					ARCHS=
 | 
				
			||||||
 | 
					prefix=${prefix-/usr/local}
 | 
				
			||||||
 | 
					exec_prefix=${exec_prefix-'${prefix}'}
 | 
				
			||||||
 | 
					libdir=${libdir-'${exec_prefix}/lib'}
 | 
				
			||||||
 | 
					sharedlibdir=${sharedlibdir-'${libdir}'}
 | 
				
			||||||
 | 
					includedir=${includedir-'${prefix}/include'}
 | 
				
			||||||
 | 
					mandir=${mandir-'${prefix}/share/man'}
 | 
				
			||||||
 | 
					shared_ext='.so'
 | 
				
			||||||
 | 
					shared=1
 | 
				
			||||||
 | 
					solo=0
 | 
				
			||||||
 | 
					cover=0
 | 
				
			||||||
 | 
					zprefix=0
 | 
				
			||||||
 | 
					zconst=0
 | 
				
			||||||
 | 
					build64=0
 | 
				
			||||||
 | 
					gcc=0
 | 
				
			||||||
 | 
					warn=0
 | 
				
			||||||
 | 
					debug=0
 | 
				
			||||||
 | 
					address=0
 | 
				
			||||||
 | 
					memory=0
 | 
				
			||||||
 | 
					old_cc="$CC"
 | 
				
			||||||
 | 
					old_cflags="$CFLAGS"
 | 
				
			||||||
 | 
					OBJC='$(OBJZ) $(OBJG)'
 | 
				
			||||||
 | 
					PIC_OBJC='$(PIC_OBJZ) $(PIC_OBJG)'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# leave this script, optionally in a bad way
 | 
				
			||||||
 | 
					leave()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  if test "$*" != "0"; then
 | 
				
			||||||
 | 
					    echo "** $0 aborting." | tee -a configure.log
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  rm -rf $test.[co] $test $test$shared_ext $test.gcno $test.dSYM ./--version
 | 
				
			||||||
 | 
					  echo -------------------- >> configure.log
 | 
				
			||||||
 | 
					  echo >> configure.log
 | 
				
			||||||
 | 
					  echo >> configure.log
 | 
				
			||||||
 | 
					  exit $1
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# process command line options
 | 
				
			||||||
 | 
					while test $# -ge 1
 | 
				
			||||||
 | 
					do
 | 
				
			||||||
 | 
					case "$1" in
 | 
				
			||||||
 | 
					    -h* | --help)
 | 
				
			||||||
 | 
					      echo 'usage:' | tee -a configure.log
 | 
				
			||||||
 | 
					      echo '  configure [--const] [--zprefix] [--prefix=PREFIX]  [--eprefix=EXPREFIX]' | tee -a configure.log
 | 
				
			||||||
 | 
					      echo '    [--static] [--64] [--libdir=LIBDIR] [--sharedlibdir=LIBDIR]' | tee -a configure.log
 | 
				
			||||||
 | 
					      echo '    [--includedir=INCLUDEDIR] [--archs="-arch i386 -arch x86_64"]' | tee -a configure.log
 | 
				
			||||||
 | 
					        exit 0 ;;
 | 
				
			||||||
 | 
					    -p*=* | --prefix=*) prefix=`echo $1 | sed 's/.*=//'`; shift ;;
 | 
				
			||||||
 | 
					    -e*=* | --eprefix=*) exec_prefix=`echo $1 | sed 's/.*=//'`; shift ;;
 | 
				
			||||||
 | 
					    -l*=* | --libdir=*) libdir=`echo $1 | sed 's/.*=//'`; shift ;;
 | 
				
			||||||
 | 
					    --sharedlibdir=*) sharedlibdir=`echo $1 | sed 's/.*=//'`; shift ;;
 | 
				
			||||||
 | 
					    -i*=* | --includedir=*) includedir=`echo $1 | sed 's/.*=//'`;shift ;;
 | 
				
			||||||
 | 
					    -u*=* | --uname=*) uname=`echo $1 | sed 's/.*=//'`;shift ;;
 | 
				
			||||||
 | 
					    -p* | --prefix) prefix="$2"; shift; shift ;;
 | 
				
			||||||
 | 
					    -e* | --eprefix) exec_prefix="$2"; shift; shift ;;
 | 
				
			||||||
 | 
					    -l* | --libdir) libdir="$2"; shift; shift ;;
 | 
				
			||||||
 | 
					    -i* | --includedir) includedir="$2"; shift; shift ;;
 | 
				
			||||||
 | 
					    -s* | --shared | --enable-shared) shared=1; shift ;;
 | 
				
			||||||
 | 
					    -t | --static) shared=0; shift ;;
 | 
				
			||||||
 | 
					    --solo) solo=1; shift ;;
 | 
				
			||||||
 | 
					    --cover) cover=1; shift ;;
 | 
				
			||||||
 | 
					    -z* | --zprefix) zprefix=1; shift ;;
 | 
				
			||||||
 | 
					    -6* | --64) build64=1; shift ;;
 | 
				
			||||||
 | 
					    -a*=* | --archs=*) ARCHS=`echo $1 | sed 's/.*=//'`; shift ;;
 | 
				
			||||||
 | 
					    --sysconfdir=*) echo "ignored option: --sysconfdir" | tee -a configure.log; shift ;;
 | 
				
			||||||
 | 
					    --localstatedir=*) echo "ignored option: --localstatedir" | tee -a configure.log; shift ;;
 | 
				
			||||||
 | 
					    -c* | --const) zconst=1; shift ;;
 | 
				
			||||||
 | 
					    -w* | --warn) warn=1; shift ;;
 | 
				
			||||||
 | 
					    -d* | --debug) debug=1; shift ;;
 | 
				
			||||||
 | 
					    --sanitize) address=1; shift ;;
 | 
				
			||||||
 | 
					    --address) address=1; shift ;;
 | 
				
			||||||
 | 
					    --memory) memory=1; shift ;;
 | 
				
			||||||
 | 
					    *)
 | 
				
			||||||
 | 
					      echo "unknown option: $1" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "$0 --help for help" | tee -a configure.log
 | 
				
			||||||
 | 
					      leave 1;;
 | 
				
			||||||
 | 
					    esac
 | 
				
			||||||
 | 
					done
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# temporary file name
 | 
				
			||||||
 | 
					test=ztest$$
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# put arguments in log, also put test file in log if used in arguments
 | 
				
			||||||
 | 
					show()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  case "$*" in
 | 
				
			||||||
 | 
					    *$test.c*)
 | 
				
			||||||
 | 
					      echo === $test.c === >> configure.log
 | 
				
			||||||
 | 
					      cat $test.c >> configure.log
 | 
				
			||||||
 | 
					      echo === >> configure.log;;
 | 
				
			||||||
 | 
					  esac
 | 
				
			||||||
 | 
					  echo $* >> configure.log
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# check for gcc vs. cc and set compile and link flags based on the system identified by uname
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					extern int getchar();
 | 
				
			||||||
 | 
					int hello() {return getchar();}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if test -z "$CC"; then
 | 
				
			||||||
 | 
					  echo Checking for ${CROSS_PREFIX}gcc... | tee -a configure.log
 | 
				
			||||||
 | 
					  if ${CROSS_PREFIX}gcc -v >/dev/null 2>&1; then
 | 
				
			||||||
 | 
					    cc=${CROSS_PREFIX}gcc
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    cc=${CROSS_PREFIX}cc
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  cc=${CC}
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					case "$cc" in
 | 
				
			||||||
 | 
					  *gcc*) gcc=1 ;;
 | 
				
			||||||
 | 
					  *clang*) gcc=1 ;;
 | 
				
			||||||
 | 
					esac
 | 
				
			||||||
 | 
					case `$cc -v 2>&1` in
 | 
				
			||||||
 | 
					  *gcc*) gcc=1 ;;
 | 
				
			||||||
 | 
					  *clang*) gcc=1 ;;
 | 
				
			||||||
 | 
					esac
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					show $cc -c $test.c
 | 
				
			||||||
 | 
					if test "$gcc" -eq 1 && ($cc -c $test.c) >> configure.log 2>&1; then
 | 
				
			||||||
 | 
					  echo ... using gcc >> configure.log
 | 
				
			||||||
 | 
					  CC="$cc"
 | 
				
			||||||
 | 
					  CFLAGS="${CFLAGS--O3}"
 | 
				
			||||||
 | 
					  SFLAGS="${CFLAGS--O3} -fPIC"
 | 
				
			||||||
 | 
					  if test "$ARCHS"; then
 | 
				
			||||||
 | 
					    CFLAGS="${CFLAGS} ${ARCHS}"
 | 
				
			||||||
 | 
					    LDFLAGS="${LDFLAGS} ${ARCHS}"
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if test $build64 -eq 1; then
 | 
				
			||||||
 | 
					    CFLAGS="${CFLAGS} -m64"
 | 
				
			||||||
 | 
					    SFLAGS="${SFLAGS} -m64"
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if test "$warn" -eq 1; then
 | 
				
			||||||
 | 
					    if test "$zconst" -eq 1; then
 | 
				
			||||||
 | 
					      CFLAGS="${CFLAGS} -Wall -Wextra -Wcast-qual -DZLIB_CONST"
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					      CFLAGS="${CFLAGS} -Wall -Wextra"
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if test $address -eq 1; then
 | 
				
			||||||
 | 
					    CFLAGS="${CFLAGS} -g -fsanitize=address -fno-omit-frame-pointer"
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if test $memory -eq 1; then
 | 
				
			||||||
 | 
					    CFLAGS="${CFLAGS} -g -fsanitize=memory -fno-omit-frame-pointer"
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if test $debug -eq 1; then
 | 
				
			||||||
 | 
					    CFLAGS="${CFLAGS} -DZLIB_DEBUG"
 | 
				
			||||||
 | 
					    SFLAGS="${SFLAGS} -DZLIB_DEBUG"
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if test -z "$uname"; then
 | 
				
			||||||
 | 
					    uname=`(uname -s || echo unknown) 2>/dev/null`
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  case "$uname" in
 | 
				
			||||||
 | 
					  Linux* | linux* | *-linux* | GNU | GNU/* | solaris*)
 | 
				
			||||||
 | 
					        case "$mname" in
 | 
				
			||||||
 | 
					        *sparc*)
 | 
				
			||||||
 | 
					            LDFLAGS="${LDFLAGS} -Wl,--no-warn-rwx-segments" ;;
 | 
				
			||||||
 | 
					        esac
 | 
				
			||||||
 | 
					        LDSHARED=${LDSHARED-"$cc -shared -Wl,-soname,libz.so.1,--version-script,${SRCDIR}zlib.map"} ;;
 | 
				
			||||||
 | 
					  *BSD | *bsd* | DragonFly)
 | 
				
			||||||
 | 
					        LDSHARED=${LDSHARED-"$cc -shared -Wl,-soname,libz.so.1,--version-script,${SRCDIR}zlib.map"}
 | 
				
			||||||
 | 
					        LDCONFIG="ldconfig -m" ;;
 | 
				
			||||||
 | 
					  CYGWIN* | Cygwin* | cygwin* | *-cygwin* | OS/2*)
 | 
				
			||||||
 | 
					        EXE='.exe' ;;
 | 
				
			||||||
 | 
					  MINGW* | mingw* | *-mingw*)
 | 
				
			||||||
 | 
					        rm -f $test.[co] $test $test$shared_ext
 | 
				
			||||||
 | 
					        echo "If this doesn't work for you, try win32/Makefile.gcc." | tee -a configure.log
 | 
				
			||||||
 | 
					        LDSHARED=${LDSHARED-"$cc -shared"}
 | 
				
			||||||
 | 
					        LDSHAREDLIBC=""
 | 
				
			||||||
 | 
					        EXE='.exe' ;;
 | 
				
			||||||
 | 
					  QNX*) # This is for QNX6. I suppose that the QNX rule below is for QNX2,QNX4
 | 
				
			||||||
 | 
					        # (alain.bonnefoy@icbt.com)
 | 
				
			||||||
 | 
					        LDSHARED=${LDSHARED-"$cc -shared -Wl,-hlibz.so.1"} ;;
 | 
				
			||||||
 | 
					  HP-UX*)
 | 
				
			||||||
 | 
					        LDSHARED=${LDSHARED-"$cc -shared $SFLAGS"}
 | 
				
			||||||
 | 
					        case `(uname -m || echo unknown) 2>/dev/null` in
 | 
				
			||||||
 | 
					        ia64)
 | 
				
			||||||
 | 
					            shared_ext='.so'
 | 
				
			||||||
 | 
					            SHAREDLIB='libz.so' ;;
 | 
				
			||||||
 | 
					        *)
 | 
				
			||||||
 | 
					            shared_ext='.sl'
 | 
				
			||||||
 | 
					            SHAREDLIB='libz.sl' ;;
 | 
				
			||||||
 | 
					        esac ;;
 | 
				
			||||||
 | 
					  AIX*)
 | 
				
			||||||
 | 
					        LDFLAGS="${LDFLAGS} -Wl,-brtl" ;;
 | 
				
			||||||
 | 
					  Darwin* | darwin* | *-darwin*)
 | 
				
			||||||
 | 
					        shared_ext='.dylib'
 | 
				
			||||||
 | 
					        SHAREDLIB=libz$shared_ext
 | 
				
			||||||
 | 
					        SHAREDLIBV=libz.$VER$shared_ext
 | 
				
			||||||
 | 
					        SHAREDLIBM=libz.$VER1$shared_ext
 | 
				
			||||||
 | 
					        LDSHARED=${LDSHARED-"$cc -dynamiclib -install_name $libdir/$SHAREDLIBM -compatibility_version $VER1 -current_version $VER3"}
 | 
				
			||||||
 | 
					        if "${CROSS_PREFIX}libtool" -V 2>&1 | grep Apple > /dev/null; then
 | 
				
			||||||
 | 
					            AR="${CROSS_PREFIX}libtool"
 | 
				
			||||||
 | 
					        elif libtool -V 2>&1 | grep Apple > /dev/null; then
 | 
				
			||||||
 | 
					            AR="libtool"
 | 
				
			||||||
 | 
					        else
 | 
				
			||||||
 | 
					            AR="/usr/bin/libtool"
 | 
				
			||||||
 | 
					        fi
 | 
				
			||||||
 | 
					        ARFLAGS="-o" ;;
 | 
				
			||||||
 | 
					  *)
 | 
				
			||||||
 | 
					        LDSHARED=${LDSHARED-"$cc -shared"} ;;
 | 
				
			||||||
 | 
					  esac
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  # find system name and corresponding cc options
 | 
				
			||||||
 | 
					  CC=${CC-cc}
 | 
				
			||||||
 | 
					  gcc=0
 | 
				
			||||||
 | 
					  echo ... using $CC >> configure.log
 | 
				
			||||||
 | 
					  if test -z "$uname"; then
 | 
				
			||||||
 | 
					    uname=`(uname -sr || echo unknown) 2>/dev/null`
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  case "$uname" in
 | 
				
			||||||
 | 
					  HP-UX*)    SFLAGS=${CFLAGS-"-O +z"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O"}
 | 
				
			||||||
 | 
					#            LDSHARED=${LDSHARED-"ld -b +vnocompatwarnings"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"ld -b"}
 | 
				
			||||||
 | 
					         case `(uname -m || echo unknown) 2>/dev/null` in
 | 
				
			||||||
 | 
					         ia64)
 | 
				
			||||||
 | 
					             shared_ext='.so'
 | 
				
			||||||
 | 
					             SHAREDLIB='libz.so' ;;
 | 
				
			||||||
 | 
					         *)
 | 
				
			||||||
 | 
					             shared_ext='.sl'
 | 
				
			||||||
 | 
					             SHAREDLIB='libz.sl' ;;
 | 
				
			||||||
 | 
					         esac ;;
 | 
				
			||||||
 | 
					  IRIX*)     SFLAGS=${CFLAGS-"-ansi -O2 -rpath ."}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-ansi -O2"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -shared -Wl,-soname,libz.so.1"} ;;
 | 
				
			||||||
 | 
					  OSF1\ V4*) SFLAGS=${CFLAGS-"-O -std1"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O -std1"}
 | 
				
			||||||
 | 
					             LDFLAGS="${LDFLAGS} -Wl,-rpath,."
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -shared  -Wl,-soname,libz.so -Wl,-msym -Wl,-rpath,$(libdir) -Wl,-set_version,${VER}:1.0"} ;;
 | 
				
			||||||
 | 
					  OSF1*)     SFLAGS=${CFLAGS-"-O -std1"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O -std1"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -shared -Wl,-soname,libz.so.1"} ;;
 | 
				
			||||||
 | 
					  QNX*)      SFLAGS=${CFLAGS-"-4 -O"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-4 -O"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc"}
 | 
				
			||||||
 | 
					             RANLIB=${RANLIB-"true"}
 | 
				
			||||||
 | 
					             AR="cc"
 | 
				
			||||||
 | 
					             ARFLAGS="-A" ;;
 | 
				
			||||||
 | 
					  SCO_SV\ 3.2*) SFLAGS=${CFLAGS-"-O3 -dy -KPIC "}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O3"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -dy -KPIC -G"} ;;
 | 
				
			||||||
 | 
					  SunOS\ 5* | solaris*)
 | 
				
			||||||
 | 
					         LDSHARED=${LDSHARED-"cc -G -h libz$shared_ext.$VER1"}
 | 
				
			||||||
 | 
					         SFLAGS=${CFLAGS-"-fast -KPIC"}
 | 
				
			||||||
 | 
					         CFLAGS=${CFLAGS-"-fast"}
 | 
				
			||||||
 | 
					         if test $build64 -eq 1; then
 | 
				
			||||||
 | 
					             # old versions of SunPRO/Workshop/Studio don't support -m64,
 | 
				
			||||||
 | 
					             # but newer ones do.  Check for it.
 | 
				
			||||||
 | 
					             flag64=`$CC -flags | egrep -- '^-m64'`
 | 
				
			||||||
 | 
					             if test x"$flag64" != x"" ; then
 | 
				
			||||||
 | 
					                 CFLAGS="${CFLAGS} -m64"
 | 
				
			||||||
 | 
					                 SFLAGS="${SFLAGS} -m64"
 | 
				
			||||||
 | 
					             else
 | 
				
			||||||
 | 
					                 case `(uname -m || echo unknown) 2>/dev/null` in
 | 
				
			||||||
 | 
					                   i86*)
 | 
				
			||||||
 | 
					                     SFLAGS="$SFLAGS -xarch=amd64"
 | 
				
			||||||
 | 
					                     CFLAGS="$CFLAGS -xarch=amd64" ;;
 | 
				
			||||||
 | 
					                   *)
 | 
				
			||||||
 | 
					                     SFLAGS="$SFLAGS -xarch=v9"
 | 
				
			||||||
 | 
					                     CFLAGS="$CFLAGS -xarch=v9" ;;
 | 
				
			||||||
 | 
					                 esac
 | 
				
			||||||
 | 
					             fi
 | 
				
			||||||
 | 
					         fi
 | 
				
			||||||
 | 
					         if test -n "$ZINC"; then
 | 
				
			||||||
 | 
					             ZINC='-I- -I. -I$(SRCDIR)'
 | 
				
			||||||
 | 
					         fi
 | 
				
			||||||
 | 
					         ;;
 | 
				
			||||||
 | 
					  SunOS\ 4*) SFLAGS=${CFLAGS-"-O2 -PIC"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O2"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"ld"} ;;
 | 
				
			||||||
 | 
					  SunStudio\ 9*) SFLAGS=${CFLAGS-"-fast -xcode=pic32 -xtarget=ultra3 -xarch=v9b"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-fast -xtarget=ultra3 -xarch=v9b"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -xarch=v9b"} ;;
 | 
				
			||||||
 | 
					  UNIX_System_V\ 4.2.0)
 | 
				
			||||||
 | 
					             SFLAGS=${CFLAGS-"-KPIC -O"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -G"} ;;
 | 
				
			||||||
 | 
					  UNIX_SV\ 4.2MP)
 | 
				
			||||||
 | 
					             SFLAGS=${CFLAGS-"-Kconform_pic -O"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -G"} ;;
 | 
				
			||||||
 | 
					  OpenUNIX\ 5)
 | 
				
			||||||
 | 
					             SFLAGS=${CFLAGS-"-KPIC -O"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -G"} ;;
 | 
				
			||||||
 | 
					  AIX*)  # Courtesy of dbakker@arrayasolutions.com
 | 
				
			||||||
 | 
					             SFLAGS=${CFLAGS-"-O -qmaxmem=8192"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O -qmaxmem=8192"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"xlc -G"} ;;
 | 
				
			||||||
 | 
					  # send working options for other systems to zlib@gzip.org
 | 
				
			||||||
 | 
					  *)         SFLAGS=${CFLAGS-"-O"}
 | 
				
			||||||
 | 
					             CFLAGS=${CFLAGS-"-O"}
 | 
				
			||||||
 | 
					             LDSHARED=${LDSHARED-"cc -shared"} ;;
 | 
				
			||||||
 | 
					  esac
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# destination names for shared library if not defined above
 | 
				
			||||||
 | 
					SHAREDLIB=${SHAREDLIB-"libz$shared_ext"}
 | 
				
			||||||
 | 
					SHAREDLIBV=${SHAREDLIBV-"libz$shared_ext.$VER"}
 | 
				
			||||||
 | 
					SHAREDLIBM=${SHAREDLIBM-"libz$shared_ext.$VER1"}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# define functions for testing compiler and library characteristics and logging the results
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#error error
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if ($CC -c $CFLAGS $test.c) 2>/dev/null; then
 | 
				
			||||||
 | 
					  try()
 | 
				
			||||||
 | 
					  {
 | 
				
			||||||
 | 
					    show $*
 | 
				
			||||||
 | 
					    test "`( $* ) 2>&1 | tee -a configure.log`" = ""
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  echo - using any output from compiler to indicate an error >> configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  try()
 | 
				
			||||||
 | 
					  {
 | 
				
			||||||
 | 
					    show $*
 | 
				
			||||||
 | 
					    got=`( $* ) 2>&1`
 | 
				
			||||||
 | 
					    ret=$?
 | 
				
			||||||
 | 
					    if test "$got" != ""; then
 | 
				
			||||||
 | 
					      printf "%s\n" "$got" >> configure.log
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					    if test $ret -ne 0; then
 | 
				
			||||||
 | 
					      echo "(exit code "$ret")" >> configure.log
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					    return $ret
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					tryboth()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  show $*
 | 
				
			||||||
 | 
					  got=`( $* ) 2>&1`
 | 
				
			||||||
 | 
					  ret=$?
 | 
				
			||||||
 | 
					  if test "$got" != ""; then
 | 
				
			||||||
 | 
					    printf "%s\n" "$got" >> configure.log
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if test $ret -ne 0; then
 | 
				
			||||||
 | 
					    echo "(exit code "$ret")" >> configure.log
 | 
				
			||||||
 | 
					    return $ret
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  test "$got" = ""
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					cat > $test.c << EOF
 | 
				
			||||||
 | 
					int foo() { return 0; }
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					echo "Checking for obsessive-compulsive compiler options..." >> configure.log
 | 
				
			||||||
 | 
					if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					  :
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  echo "Compiler error reporting is too harsh for $0 (perhaps remove -Werror)." | tee -a configure.log
 | 
				
			||||||
 | 
					  leave 1
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# see if shared library build supported
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					extern int getchar();
 | 
				
			||||||
 | 
					int hello() {return getchar();}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if test $shared -eq 1; then
 | 
				
			||||||
 | 
					  echo Checking for shared library support... | tee -a configure.log
 | 
				
			||||||
 | 
					  # we must test in two steps (cc then ld), required at least on SunOS 4.x
 | 
				
			||||||
 | 
					  if try $CC -c $SFLAGS $test.c &&
 | 
				
			||||||
 | 
					     try $LDSHARED $SFLAGS -o $test$shared_ext $test.o; then
 | 
				
			||||||
 | 
					    echo Building shared library $SHAREDLIBV with $CC. | tee -a configure.log
 | 
				
			||||||
 | 
					  elif test -z "$old_cc" -a -z "$old_cflags"; then
 | 
				
			||||||
 | 
					    echo No shared library support. | tee -a configure.log
 | 
				
			||||||
 | 
					    shared=0;
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    echo 'No shared library support; try without defining CC and CFLAGS' | tee -a configure.log
 | 
				
			||||||
 | 
					    shared=0;
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					if test $shared -eq 0; then
 | 
				
			||||||
 | 
					  LDSHARED="$CC"
 | 
				
			||||||
 | 
					  ALL="static"
 | 
				
			||||||
 | 
					  TEST="all teststatic"
 | 
				
			||||||
 | 
					  SHAREDLIB=""
 | 
				
			||||||
 | 
					  SHAREDLIBV=""
 | 
				
			||||||
 | 
					  SHAREDLIBM=""
 | 
				
			||||||
 | 
					  echo Building static library $STATICLIB version $VER with $CC. | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  ALL="static shared"
 | 
				
			||||||
 | 
					  TEST="all teststatic testshared"
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# check for size_t
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <stdlib.h>
 | 
				
			||||||
 | 
					size_t dummy = 0;
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					  echo "Checking for size_t... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  echo "Checking for size_t... No." | tee -a configure.log
 | 
				
			||||||
 | 
					  # find a size_t integer type
 | 
				
			||||||
 | 
					  # check for long long
 | 
				
			||||||
 | 
					  cat > $test.c << EOF
 | 
				
			||||||
 | 
					long long dummy = 0;
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					  if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					    echo "Checking for long long... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					    cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					int main(void) {
 | 
				
			||||||
 | 
					    if (sizeof(void *) <= sizeof(int)) puts("int");
 | 
				
			||||||
 | 
					    else if (sizeof(void *) <= sizeof(long)) puts("long");
 | 
				
			||||||
 | 
					    else puts("z_longlong");
 | 
				
			||||||
 | 
					    return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    echo "Checking for long long... No." | tee -a configure.log
 | 
				
			||||||
 | 
					    cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					int main(void) {
 | 
				
			||||||
 | 
					    if (sizeof(void *) <= sizeof(int)) puts("int");
 | 
				
			||||||
 | 
					    else puts("long");
 | 
				
			||||||
 | 
					    return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					  if try $CC $CFLAGS -o $test $test.c; then
 | 
				
			||||||
 | 
					    sizet=`./$test`
 | 
				
			||||||
 | 
					    echo "Checking for a pointer-size integer type..." $sizet"." | tee -a configure.log
 | 
				
			||||||
 | 
					    CFLAGS="${CFLAGS} -DNO_SIZE_T=${sizet}"
 | 
				
			||||||
 | 
					    SFLAGS="${SFLAGS} -DNO_SIZE_T=${sizet}"
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    echo "Checking for a pointer-size integer type... not found." | tee -a configure.log
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# check for large file support, and if none, check for fseeko()
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <sys/types.h>
 | 
				
			||||||
 | 
					off64_t dummy = 0;
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if try $CC -c $CFLAGS -D_LARGEFILE64_SOURCE=1 $test.c; then
 | 
				
			||||||
 | 
					  CFLAGS="${CFLAGS} -D_LARGEFILE64_SOURCE=1"
 | 
				
			||||||
 | 
					  SFLAGS="${SFLAGS} -D_LARGEFILE64_SOURCE=1"
 | 
				
			||||||
 | 
					  ALL="${ALL} all64"
 | 
				
			||||||
 | 
					  TEST="${TEST} test64"
 | 
				
			||||||
 | 
					  echo "Checking for off64_t... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					  echo "Checking for fseeko... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  echo "Checking for off64_t... No." | tee -a configure.log
 | 
				
			||||||
 | 
					  echo >> configure.log
 | 
				
			||||||
 | 
					  cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					int main(void) {
 | 
				
			||||||
 | 
					  fseeko(NULL, 0, 0);
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					  if try $CC $CFLAGS -o $test $test.c; then
 | 
				
			||||||
 | 
					    echo "Checking for fseeko... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    CFLAGS="${CFLAGS} -DNO_FSEEKO"
 | 
				
			||||||
 | 
					    SFLAGS="${SFLAGS} -DNO_FSEEKO"
 | 
				
			||||||
 | 
					    echo "Checking for fseeko... No." | tee -a configure.log
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# check for strerror() for use by gz* functions
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <string.h>
 | 
				
			||||||
 | 
					#include <errno.h>
 | 
				
			||||||
 | 
					int main() { return strlen(strerror(errno)); }
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if try $CC $CFLAGS -o $test $test.c; then
 | 
				
			||||||
 | 
					  echo "Checking for strerror... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  CFLAGS="${CFLAGS} -DNO_STRERROR"
 | 
				
			||||||
 | 
					  SFLAGS="${SFLAGS} -DNO_STRERROR"
 | 
				
			||||||
 | 
					  echo "Checking for strerror... No." | tee -a configure.log
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# copy clean zconf.h for subsequent edits
 | 
				
			||||||
 | 
					cp -p ${SRCDIR}zconf.h.in zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# check for unistd.h and save result in zconf.h
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <unistd.h>
 | 
				
			||||||
 | 
					int main() { return 0; }
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					  sed < zconf.h "/^#ifdef HAVE_UNISTD_H.* may be/s/def HAVE_UNISTD_H\(.*\) may be/ 1\1 was/" > zconf.temp.h
 | 
				
			||||||
 | 
					  mv zconf.temp.h zconf.h
 | 
				
			||||||
 | 
					  echo "Checking for unistd.h... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  echo "Checking for unistd.h... No." | tee -a configure.log
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# check for stdarg.h and save result in zconf.h
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdarg.h>
 | 
				
			||||||
 | 
					int main() { return 0; }
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					  sed < zconf.h "/^#ifdef HAVE_STDARG_H.* may be/s/def HAVE_STDARG_H\(.*\) may be/ 1\1 was/" > zconf.temp.h
 | 
				
			||||||
 | 
					  mv zconf.temp.h zconf.h
 | 
				
			||||||
 | 
					  echo "Checking for stdarg.h... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  echo "Checking for stdarg.h... No." | tee -a configure.log
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# if the z_ prefix was requested, save that in zconf.h
 | 
				
			||||||
 | 
					if test $zprefix -eq 1; then
 | 
				
			||||||
 | 
					  sed < zconf.h "/#ifdef Z_PREFIX.* may be/s/def Z_PREFIX\(.*\) may be/ 1\1 was/" > zconf.temp.h
 | 
				
			||||||
 | 
					  mv zconf.temp.h zconf.h
 | 
				
			||||||
 | 
					  echo >> configure.log
 | 
				
			||||||
 | 
					  echo "Using z_ prefix on all symbols." | tee -a configure.log
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# if --solo compilation was requested, save that in zconf.h and remove gz stuff from object lists
 | 
				
			||||||
 | 
					if test $solo -eq 1; then
 | 
				
			||||||
 | 
					  sed '/#define ZCONF_H/a\
 | 
				
			||||||
 | 
					#define Z_SOLO
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					' < zconf.h > zconf.temp.h
 | 
				
			||||||
 | 
					  mv zconf.temp.h zconf.h
 | 
				
			||||||
 | 
					OBJC='$(OBJZ)'
 | 
				
			||||||
 | 
					PIC_OBJC='$(PIC_OBJZ)'
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# if code coverage testing was requested, use older gcc if defined, e.g. "gcc-4.2" on Mac OS X
 | 
				
			||||||
 | 
					if test $cover -eq 1; then
 | 
				
			||||||
 | 
					  CFLAGS="${CFLAGS} -fprofile-arcs -ftest-coverage"
 | 
				
			||||||
 | 
					  if test -n "$GCC_CLASSIC"; then
 | 
				
			||||||
 | 
					    CC=$GCC_CLASSIC
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# conduct a series of tests to resolve eight possible cases of using "vs" or "s" printf functions
 | 
				
			||||||
 | 
					# (using stdarg or not), with or without "n" (proving size of buffer), and with or without a
 | 
				
			||||||
 | 
					# return value.  The most secure result is vsnprintf() with a return value.  snprintf() with a
 | 
				
			||||||
 | 
					# return value is secure as well, but then gzprintf() will be limited to 20 arguments.
 | 
				
			||||||
 | 
					cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <stdarg.h>
 | 
				
			||||||
 | 
					#include "zconf.h"
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					#ifndef STDC
 | 
				
			||||||
 | 
					  choke me
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					  echo "Checking whether to use vs[n]printf() or s[n]printf()... using vs[n]printf()." | tee -a configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  echo >> configure.log
 | 
				
			||||||
 | 
					  cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <stdarg.h>
 | 
				
			||||||
 | 
					int mytest(const char *fmt, ...)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  char buf[20];
 | 
				
			||||||
 | 
					  va_list ap;
 | 
				
			||||||
 | 
					  va_start(ap, fmt);
 | 
				
			||||||
 | 
					  vsnprintf(buf, sizeof(buf), fmt, ap);
 | 
				
			||||||
 | 
					  va_end(ap);
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  return (mytest("Hello%d\n", 1));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					  if try $CC $CFLAGS -o $test $test.c; then
 | 
				
			||||||
 | 
					    echo "Checking for vsnprintf() in stdio.h... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    echo >> configure.log
 | 
				
			||||||
 | 
					    cat >$test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <stdarg.h>
 | 
				
			||||||
 | 
					int mytest(const char *fmt, ...)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  int n;
 | 
				
			||||||
 | 
					  char buf[20];
 | 
				
			||||||
 | 
					  va_list ap;
 | 
				
			||||||
 | 
					  va_start(ap, fmt);
 | 
				
			||||||
 | 
					  n = vsnprintf(buf, sizeof(buf), fmt, ap);
 | 
				
			||||||
 | 
					  va_end(ap);
 | 
				
			||||||
 | 
					  return n;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  return (mytest("Hello%d\n", 1));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					      echo "Checking for return value of vsnprintf()... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					      CFLAGS="$CFLAGS -DHAS_vsnprintf_void"
 | 
				
			||||||
 | 
					      SFLAGS="$SFLAGS -DHAS_vsnprintf_void"
 | 
				
			||||||
 | 
					      echo "Checking for return value of vsnprintf()... No." | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  WARNING: apparently vsnprintf() does not return a value. zlib" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  can build but will be open to possible string-format security" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  vulnerabilities." | tee -a configure.log
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    CFLAGS="$CFLAGS -DNO_vsnprintf"
 | 
				
			||||||
 | 
					    SFLAGS="$SFLAGS -DNO_vsnprintf"
 | 
				
			||||||
 | 
					    echo "Checking for vsnprintf() in stdio.h... No." | tee -a configure.log
 | 
				
			||||||
 | 
					    echo "  WARNING: vsnprintf() not found, falling back to vsprintf(). zlib" | tee -a configure.log
 | 
				
			||||||
 | 
					    echo "  can build but will be open to possible buffer-overflow security" | tee -a configure.log
 | 
				
			||||||
 | 
					    echo "  vulnerabilities." | tee -a configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    echo >> configure.log
 | 
				
			||||||
 | 
					    cat >$test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <stdarg.h>
 | 
				
			||||||
 | 
					int mytest(const char *fmt, ...)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  int n;
 | 
				
			||||||
 | 
					  char buf[20];
 | 
				
			||||||
 | 
					  va_list ap;
 | 
				
			||||||
 | 
					  va_start(ap, fmt);
 | 
				
			||||||
 | 
					  n = vsprintf(buf, fmt, ap);
 | 
				
			||||||
 | 
					  va_end(ap);
 | 
				
			||||||
 | 
					  return n;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  return (mytest("Hello%d\n", 1));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					      echo "Checking for return value of vsprintf()... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					      CFLAGS="$CFLAGS -DHAS_vsprintf_void"
 | 
				
			||||||
 | 
					      SFLAGS="$SFLAGS -DHAS_vsprintf_void"
 | 
				
			||||||
 | 
					      echo "Checking for return value of vsprintf()... No." | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  WARNING: apparently vsprintf() does not return a value. zlib" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  can build but will be open to possible string-format security" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  vulnerabilities." | tee -a configure.log
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					else
 | 
				
			||||||
 | 
					  echo "Checking whether to use vs[n]printf() or s[n]printf()... using s[n]printf()." | tee -a configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  echo >> configure.log
 | 
				
			||||||
 | 
					  cat >$test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					int mytest()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  char buf[20];
 | 
				
			||||||
 | 
					  snprintf(buf, sizeof(buf), "%s", "foo");
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  return (mytest());
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  if try $CC $CFLAGS -o $test $test.c; then
 | 
				
			||||||
 | 
					    echo "Checking for snprintf() in stdio.h... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    echo >> configure.log
 | 
				
			||||||
 | 
					    cat >$test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					int mytest()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  char buf[20];
 | 
				
			||||||
 | 
					  return snprintf(buf, sizeof(buf), "%s", "foo");
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  return (mytest());
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					      echo "Checking for return value of snprintf()... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					      CFLAGS="$CFLAGS -DHAS_snprintf_void"
 | 
				
			||||||
 | 
					      SFLAGS="$SFLAGS -DHAS_snprintf_void"
 | 
				
			||||||
 | 
					      echo "Checking for return value of snprintf()... No." | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  WARNING: apparently snprintf() does not return a value. zlib" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  can build but will be open to possible string-format security" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  vulnerabilities." | tee -a configure.log
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    CFLAGS="$CFLAGS -DNO_snprintf"
 | 
				
			||||||
 | 
					    SFLAGS="$SFLAGS -DNO_snprintf"
 | 
				
			||||||
 | 
					    echo "Checking for snprintf() in stdio.h... No." | tee -a configure.log
 | 
				
			||||||
 | 
					    echo "  WARNING: snprintf() not found, falling back to sprintf(). zlib" | tee -a configure.log
 | 
				
			||||||
 | 
					    echo "  can build but will be open to possible buffer-overflow security" | tee -a configure.log
 | 
				
			||||||
 | 
					    echo "  vulnerabilities." | tee -a configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    echo >> configure.log
 | 
				
			||||||
 | 
					    cat >$test.c <<EOF
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					int mytest()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  char buf[20];
 | 
				
			||||||
 | 
					  return sprintf(buf, "%s", "foo");
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  return (mytest());
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if try $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					      echo "Checking for return value of sprintf()... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					      CFLAGS="$CFLAGS -DHAS_sprintf_void"
 | 
				
			||||||
 | 
					      SFLAGS="$SFLAGS -DHAS_sprintf_void"
 | 
				
			||||||
 | 
					      echo "Checking for return value of sprintf()... No." | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  WARNING: apparently sprintf() does not return a value. zlib" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  can build but will be open to possible string-format security" | tee -a configure.log
 | 
				
			||||||
 | 
					      echo "  vulnerabilities." | tee -a configure.log
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# see if we can hide zlib internal symbols that are linked between separate source files
 | 
				
			||||||
 | 
					if test "$gcc" -eq 1; then
 | 
				
			||||||
 | 
					  echo >> configure.log
 | 
				
			||||||
 | 
					  cat > $test.c <<EOF
 | 
				
			||||||
 | 
					#define ZLIB_INTERNAL __attribute__((visibility ("hidden")))
 | 
				
			||||||
 | 
					int ZLIB_INTERNAL foo;
 | 
				
			||||||
 | 
					int main()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					EOF
 | 
				
			||||||
 | 
					  if tryboth $CC -c $CFLAGS $test.c; then
 | 
				
			||||||
 | 
					    CFLAGS="$CFLAGS -DHAVE_HIDDEN"
 | 
				
			||||||
 | 
					    SFLAGS="$SFLAGS -DHAVE_HIDDEN"
 | 
				
			||||||
 | 
					    echo "Checking for attribute(visibility) support... Yes." | tee -a configure.log
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    echo "Checking for attribute(visibility) support... No." | tee -a configure.log
 | 
				
			||||||
 | 
					  fi
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# show the results in the log
 | 
				
			||||||
 | 
					echo >> configure.log
 | 
				
			||||||
 | 
					echo ALL = $ALL >> configure.log
 | 
				
			||||||
 | 
					echo AR = $AR >> configure.log
 | 
				
			||||||
 | 
					echo ARFLAGS = $ARFLAGS >> configure.log
 | 
				
			||||||
 | 
					echo CC = $CC >> configure.log
 | 
				
			||||||
 | 
					echo CFLAGS = $CFLAGS >> configure.log
 | 
				
			||||||
 | 
					echo CPP = $CPP >> configure.log
 | 
				
			||||||
 | 
					echo EXE = $EXE >> configure.log
 | 
				
			||||||
 | 
					echo LDCONFIG = $LDCONFIG >> configure.log
 | 
				
			||||||
 | 
					echo LDFLAGS = $LDFLAGS >> configure.log
 | 
				
			||||||
 | 
					echo LDSHARED = $LDSHARED >> configure.log
 | 
				
			||||||
 | 
					echo LDSHAREDLIBC = $LDSHAREDLIBC >> configure.log
 | 
				
			||||||
 | 
					echo OBJC = $OBJC >> configure.log
 | 
				
			||||||
 | 
					echo PIC_OBJC = $PIC_OBJC >> configure.log
 | 
				
			||||||
 | 
					echo RANLIB = $RANLIB >> configure.log
 | 
				
			||||||
 | 
					echo SFLAGS = $SFLAGS >> configure.log
 | 
				
			||||||
 | 
					echo SHAREDLIB = $SHAREDLIB >> configure.log
 | 
				
			||||||
 | 
					echo SHAREDLIBM = $SHAREDLIBM >> configure.log
 | 
				
			||||||
 | 
					echo SHAREDLIBV = $SHAREDLIBV >> configure.log
 | 
				
			||||||
 | 
					echo STATICLIB = $STATICLIB >> configure.log
 | 
				
			||||||
 | 
					echo TEST = $TEST >> configure.log
 | 
				
			||||||
 | 
					echo VER = $VER >> configure.log
 | 
				
			||||||
 | 
					echo SRCDIR = $SRCDIR >> configure.log
 | 
				
			||||||
 | 
					echo exec_prefix = $exec_prefix >> configure.log
 | 
				
			||||||
 | 
					echo includedir = $includedir >> configure.log
 | 
				
			||||||
 | 
					echo libdir = $libdir >> configure.log
 | 
				
			||||||
 | 
					echo mandir = $mandir >> configure.log
 | 
				
			||||||
 | 
					echo prefix = $prefix >> configure.log
 | 
				
			||||||
 | 
					echo sharedlibdir = $sharedlibdir >> configure.log
 | 
				
			||||||
 | 
					echo uname = $uname >> configure.log
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# update Makefile with the configure results
 | 
				
			||||||
 | 
					sed < ${SRCDIR}Makefile.in "
 | 
				
			||||||
 | 
					/^CC *=/s#=.*#=$CC#
 | 
				
			||||||
 | 
					/^CFLAGS *=/s#=.*#=$CFLAGS#
 | 
				
			||||||
 | 
					/^SFLAGS *=/s#=.*#=$SFLAGS#
 | 
				
			||||||
 | 
					/^LDFLAGS *=/s#=.*#=$LDFLAGS#
 | 
				
			||||||
 | 
					/^LDSHARED *=/s#=.*#=$LDSHARED#
 | 
				
			||||||
 | 
					/^CPP *=/s#=.*#=$CPP#
 | 
				
			||||||
 | 
					/^STATICLIB *=/s#=.*#=$STATICLIB#
 | 
				
			||||||
 | 
					/^SHAREDLIB *=/s#=.*#=$SHAREDLIB#
 | 
				
			||||||
 | 
					/^SHAREDLIBV *=/s#=.*#=$SHAREDLIBV#
 | 
				
			||||||
 | 
					/^SHAREDLIBM *=/s#=.*#=$SHAREDLIBM#
 | 
				
			||||||
 | 
					/^AR *=/s#=.*#=$AR#
 | 
				
			||||||
 | 
					/^ARFLAGS *=/s#=.*#=$ARFLAGS#
 | 
				
			||||||
 | 
					/^RANLIB *=/s#=.*#=$RANLIB#
 | 
				
			||||||
 | 
					/^LDCONFIG *=/s#=.*#=$LDCONFIG#
 | 
				
			||||||
 | 
					/^LDSHAREDLIBC *=/s#=.*#=$LDSHAREDLIBC#
 | 
				
			||||||
 | 
					/^EXE *=/s#=.*#=$EXE#
 | 
				
			||||||
 | 
					/^SRCDIR *=/s#=.*#=$SRCDIR#
 | 
				
			||||||
 | 
					/^ZINC *=/s#=.*#=$ZINC#
 | 
				
			||||||
 | 
					/^ZINCOUT *=/s#=.*#=$ZINCOUT#
 | 
				
			||||||
 | 
					/^prefix *=/s#=.*#=$prefix#
 | 
				
			||||||
 | 
					/^exec_prefix *=/s#=.*#=$exec_prefix#
 | 
				
			||||||
 | 
					/^libdir *=/s#=.*#=$libdir#
 | 
				
			||||||
 | 
					/^sharedlibdir *=/s#=.*#=$sharedlibdir#
 | 
				
			||||||
 | 
					/^includedir *=/s#=.*#=$includedir#
 | 
				
			||||||
 | 
					/^mandir *=/s#=.*#=$mandir#
 | 
				
			||||||
 | 
					/^OBJC *=/s#=.*#= $OBJC#
 | 
				
			||||||
 | 
					/^PIC_OBJC *=/s#=.*#= $PIC_OBJC#
 | 
				
			||||||
 | 
					/^all: */s#:.*#: $ALL#
 | 
				
			||||||
 | 
					/^test: */s#:.*#: $TEST#
 | 
				
			||||||
 | 
					" > Makefile
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# create zlib.pc with the configure results
 | 
				
			||||||
 | 
					sed < ${SRCDIR}zlib.pc.in "
 | 
				
			||||||
 | 
					/^CC *=/s#=.*#=$CC#
 | 
				
			||||||
 | 
					/^CFLAGS *=/s#=.*#=$CFLAGS#
 | 
				
			||||||
 | 
					/^CPP *=/s#=.*#=$CPP#
 | 
				
			||||||
 | 
					/^LDSHARED *=/s#=.*#=$LDSHARED#
 | 
				
			||||||
 | 
					/^STATICLIB *=/s#=.*#=$STATICLIB#
 | 
				
			||||||
 | 
					/^SHAREDLIB *=/s#=.*#=$SHAREDLIB#
 | 
				
			||||||
 | 
					/^SHAREDLIBV *=/s#=.*#=$SHAREDLIBV#
 | 
				
			||||||
 | 
					/^SHAREDLIBM *=/s#=.*#=$SHAREDLIBM#
 | 
				
			||||||
 | 
					/^AR *=/s#=.*#=$AR#
 | 
				
			||||||
 | 
					/^ARFLAGS *=/s#=.*#=$ARFLAGS#
 | 
				
			||||||
 | 
					/^RANLIB *=/s#=.*#=$RANLIB#
 | 
				
			||||||
 | 
					/^EXE *=/s#=.*#=$EXE#
 | 
				
			||||||
 | 
					/^prefix *=/s#=.*#=$prefix#
 | 
				
			||||||
 | 
					/^exec_prefix *=/s#=.*#=$exec_prefix#
 | 
				
			||||||
 | 
					/^libdir *=/s#=.*#=$libdir#
 | 
				
			||||||
 | 
					/^sharedlibdir *=/s#=.*#=$sharedlibdir#
 | 
				
			||||||
 | 
					/^includedir *=/s#=.*#=$includedir#
 | 
				
			||||||
 | 
					/^mandir *=/s#=.*#=$mandir#
 | 
				
			||||||
 | 
					/^LDFLAGS *=/s#=.*#=$LDFLAGS#
 | 
				
			||||||
 | 
					" | sed -e "
 | 
				
			||||||
 | 
					s/\@VERSION\@/$VER/g;
 | 
				
			||||||
 | 
					" > zlib.pc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# done
 | 
				
			||||||
 | 
					leave 0
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,57 @@
 | 
				
			||||||
 | 
					All files under this contrib directory are UNSUPPORTED. They were
 | 
				
			||||||
 | 
					provided by users of zlib and were not tested by the authors of zlib.
 | 
				
			||||||
 | 
					Use at your own risk. Please contact the authors of the contributions
 | 
				
			||||||
 | 
					for help about these, not the zlib authors. Thanks.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ada/        by Dmitriy Anisimkov <anisimkov@yahoo.com>
 | 
				
			||||||
 | 
					        Support for Ada
 | 
				
			||||||
 | 
					        See http://zlib-ada.sourceforge.net/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					blast/      by Mark Adler <madler@alumni.caltech.edu>
 | 
				
			||||||
 | 
					        Decompressor for output of PKWare Data Compression Library (DCL)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					delphi/     by Cosmin Truta <cosmint@cs.ubbcluj.ro>
 | 
				
			||||||
 | 
					        Support for Delphi and C++ Builder
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					dotzlib/    by Henrik Ravn <henrik@ravn.com>
 | 
				
			||||||
 | 
					        Support for Microsoft .Net and Visual C++ .Net
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gcc_gvmat64/by Gilles Vollant <info@winimage.com>
 | 
				
			||||||
 | 
					        GCC Version of x86 64-bit (AMD64 and Intel EM64t) code for x64
 | 
				
			||||||
 | 
					        assembler to replace longest_match() and inflate_fast()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					infback9/   by Mark Adler <madler@alumni.caltech.edu>
 | 
				
			||||||
 | 
					        Unsupported diffs to infback to decode the deflate64 format
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					iostream/   by Kevin Ruland <kevin@rodin.wustl.edu>
 | 
				
			||||||
 | 
					        A C++ I/O streams interface to the zlib gz* functions
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					iostream2/  by Tyge Løvset <Tyge.Lovset@cmr.no>
 | 
				
			||||||
 | 
					        Another C++ I/O streams interface
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					iostream3/  by Ludwig Schwardt <schwardt@sun.ac.za>
 | 
				
			||||||
 | 
					            and Kevin Ruland <kevin@rodin.wustl.edu>
 | 
				
			||||||
 | 
					        Yet another C++ I/O streams interface
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minizip/    by Gilles Vollant <info@winimage.com>
 | 
				
			||||||
 | 
					        Mini zip and unzip based on zlib
 | 
				
			||||||
 | 
					        Includes Zip64 support by Mathias Svensson <mathias@result42.com>
 | 
				
			||||||
 | 
					        See http://www.winimage.com/zLibDll/minizip.html
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pascal/     by Bob Dellaca <bobdl@xtra.co.nz> et al.
 | 
				
			||||||
 | 
					        Support for Pascal
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					puff/       by Mark Adler <madler@alumni.caltech.edu>
 | 
				
			||||||
 | 
					        Small, low memory usage inflate.  Also serves to provide an
 | 
				
			||||||
 | 
					        unambiguous description of the deflate format.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					testzlib/   by Gilles Vollant <info@winimage.com>
 | 
				
			||||||
 | 
					        Example of the use of zlib
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					untgz/      by Pedro A. Aranda Gutierrez <paag@tid.es>
 | 
				
			||||||
 | 
					        A very simple tar.gz file extractor using zlib
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					vstudio/    by Gilles Vollant <info@winimage.com>
 | 
				
			||||||
 | 
					        Building a minizip-enhanced zlib with Microsoft Visual Studio
 | 
				
			||||||
 | 
					        Includes vc11 from kreuzerkrieg and vc12 from davispuh
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,106 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2004 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--
 | 
				
			||||||
 | 
					--  $Id: buffer_demo.adb,v 1.3 2004/09/06 06:55:35 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  This demo program provided by Dr Steve Sangwine <sjs@essex.ac.uk>
 | 
				
			||||||
 | 
					--
 | 
				
			||||||
 | 
					--  Demonstration of a problem with Zlib-Ada (already fixed) when a buffer
 | 
				
			||||||
 | 
					--  of exactly the correct size is used for decompressed data, and the last
 | 
				
			||||||
 | 
					--  few bytes passed in to Zlib are checksum bytes.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  This program compresses a string of text, and then decompresses the
 | 
				
			||||||
 | 
					--  compressed text into a buffer of the same size as the original text.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Ada.Streams; use Ada.Streams;
 | 
				
			||||||
 | 
					with Ada.Text_IO;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with ZLib; use ZLib;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure Buffer_Demo is
 | 
				
			||||||
 | 
					   EOL  : Character renames ASCII.LF;
 | 
				
			||||||
 | 
					   Text : constant String
 | 
				
			||||||
 | 
					     := "Four score and seven years ago our fathers brought forth," & EOL &
 | 
				
			||||||
 | 
					        "upon this continent, a new nation, conceived in liberty," & EOL &
 | 
				
			||||||
 | 
					        "and dedicated to the proposition that `all men are created equal'.";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Source : Stream_Element_Array (1 .. Text'Length);
 | 
				
			||||||
 | 
					   for Source'Address use Text'Address;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					   Ada.Text_IO.Put (Text);
 | 
				
			||||||
 | 
					   Ada.Text_IO.New_Line;
 | 
				
			||||||
 | 
					   Ada.Text_IO.Put_Line
 | 
				
			||||||
 | 
					     ("Uncompressed size : " & Positive'Image (Text'Length) & " bytes");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   declare
 | 
				
			||||||
 | 
					      Compressed_Data : Stream_Element_Array (1 .. Text'Length);
 | 
				
			||||||
 | 
					      L               : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Compress : declare
 | 
				
			||||||
 | 
					         Compressor : Filter_Type;
 | 
				
			||||||
 | 
					         I : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         Deflate_Init (Compressor);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Compress the whole of T at once.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Translate (Compressor, Source, I, Compressed_Data, L, Finish);
 | 
				
			||||||
 | 
					         pragma Assert (I = Source'Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Close (Compressor);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Ada.Text_IO.Put_Line
 | 
				
			||||||
 | 
					           ("Compressed size :   "
 | 
				
			||||||
 | 
					            & Stream_Element_Offset'Image (L) & " bytes");
 | 
				
			||||||
 | 
					      end Compress;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      --  Now we decompress the data, passing short blocks of data to Zlib
 | 
				
			||||||
 | 
					      --  (because this demonstrates the problem - the last block passed will
 | 
				
			||||||
 | 
					      --  contain checksum information and there will be no output, only a
 | 
				
			||||||
 | 
					      --  check inside Zlib that the checksum is correct).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Decompress : declare
 | 
				
			||||||
 | 
					         Decompressor : Filter_Type;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Uncompressed_Data : Stream_Element_Array (1 .. Text'Length);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Block_Size : constant := 4;
 | 
				
			||||||
 | 
					         --  This makes sure that the last block contains
 | 
				
			||||||
 | 
					         --  only Adler checksum data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         P : Stream_Element_Offset := Compressed_Data'First - 1;
 | 
				
			||||||
 | 
					         O : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         Inflate_Init (Decompressor);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         loop
 | 
				
			||||||
 | 
					            Translate
 | 
				
			||||||
 | 
					              (Decompressor,
 | 
				
			||||||
 | 
					               Compressed_Data
 | 
				
			||||||
 | 
					                 (P + 1 .. Stream_Element_Offset'Min (P + Block_Size, L)),
 | 
				
			||||||
 | 
					               P,
 | 
				
			||||||
 | 
					               Uncompressed_Data
 | 
				
			||||||
 | 
					                 (Total_Out (Decompressor) + 1 .. Uncompressed_Data'Last),
 | 
				
			||||||
 | 
					               O,
 | 
				
			||||||
 | 
					               No_Flush);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					               Ada.Text_IO.Put_Line
 | 
				
			||||||
 | 
					                 ("Total in : " & Count'Image (Total_In (Decompressor)) &
 | 
				
			||||||
 | 
					                  ", out : " & Count'Image (Total_Out (Decompressor)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					               exit when P = L;
 | 
				
			||||||
 | 
					         end loop;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Ada.Text_IO.New_Line;
 | 
				
			||||||
 | 
					         Ada.Text_IO.Put_Line
 | 
				
			||||||
 | 
					           ("Decompressed text matches original text : "
 | 
				
			||||||
 | 
					             & Boolean'Image (Uncompressed_Data = Source));
 | 
				
			||||||
 | 
					      end Decompress;
 | 
				
			||||||
 | 
					   end;
 | 
				
			||||||
 | 
					end Buffer_Demo;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,156 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2003 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  Continuous test for ZLib multithreading. If the test would fail
 | 
				
			||||||
 | 
					--  we should provide thread safe allocation routines for the Z_Stream.
 | 
				
			||||||
 | 
					--
 | 
				
			||||||
 | 
					--  $Id: mtest.adb,v 1.4 2004/07/23 07:49:54 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with ZLib;
 | 
				
			||||||
 | 
					with Ada.Streams;
 | 
				
			||||||
 | 
					with Ada.Numerics.Discrete_Random;
 | 
				
			||||||
 | 
					with Ada.Text_IO;
 | 
				
			||||||
 | 
					with Ada.Exceptions;
 | 
				
			||||||
 | 
					with Ada.Task_Identification;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure MTest is
 | 
				
			||||||
 | 
					   use Ada.Streams;
 | 
				
			||||||
 | 
					   use ZLib;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Stop : Boolean := False;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   pragma Atomic (Stop);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   subtype Visible_Symbols is Stream_Element range 16#20# .. 16#7E#;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   package Random_Elements is
 | 
				
			||||||
 | 
					      new Ada.Numerics.Discrete_Random (Visible_Symbols);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   task type Test_Task;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   task body Test_Task is
 | 
				
			||||||
 | 
					      Buffer : Stream_Element_Array (1 .. 100_000);
 | 
				
			||||||
 | 
					      Gen : Random_Elements.Generator;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Buffer_First  : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Compare_First : Stream_Element_Offset;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Deflate : Filter_Type;
 | 
				
			||||||
 | 
					      Inflate : Filter_Type;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Further (Item : in Stream_Element_Array);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Read_Buffer
 | 
				
			||||||
 | 
					        (Item : out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					         Last : out Ada.Streams.Stream_Element_Offset);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      -------------
 | 
				
			||||||
 | 
					      -- Further --
 | 
				
			||||||
 | 
					      -------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Further (Item : in Stream_Element_Array) is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         procedure Compare (Item : in Stream_Element_Array);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         -------------
 | 
				
			||||||
 | 
					         -- Compare --
 | 
				
			||||||
 | 
					         -------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         procedure Compare (Item : in Stream_Element_Array) is
 | 
				
			||||||
 | 
					            Next_First : Stream_Element_Offset := Compare_First + Item'Length;
 | 
				
			||||||
 | 
					         begin
 | 
				
			||||||
 | 
					            if Buffer (Compare_First .. Next_First - 1) /= Item then
 | 
				
			||||||
 | 
					               raise Program_Error;
 | 
				
			||||||
 | 
					            end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Compare_First := Next_First;
 | 
				
			||||||
 | 
					         end Compare;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         procedure Compare_Write is new ZLib.Write (Write => Compare);
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         Compare_Write (Inflate, Item, No_Flush);
 | 
				
			||||||
 | 
					      end Further;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      -----------------
 | 
				
			||||||
 | 
					      -- Read_Buffer --
 | 
				
			||||||
 | 
					      -----------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Read_Buffer
 | 
				
			||||||
 | 
					        (Item : out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					         Last : out Ada.Streams.Stream_Element_Offset)
 | 
				
			||||||
 | 
					      is
 | 
				
			||||||
 | 
					         Buff_Diff   : Stream_Element_Offset := Buffer'Last - Buffer_First;
 | 
				
			||||||
 | 
					         Next_First : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         if Item'Length <= Buff_Diff then
 | 
				
			||||||
 | 
					            Last := Item'Last;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Next_First := Buffer_First + Item'Length;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Item := Buffer (Buffer_First .. Next_First - 1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Buffer_First := Next_First;
 | 
				
			||||||
 | 
					         else
 | 
				
			||||||
 | 
					            Last := Item'First + Buff_Diff;
 | 
				
			||||||
 | 
					            Item (Item'First .. Last) := Buffer (Buffer_First .. Buffer'Last);
 | 
				
			||||||
 | 
					            Buffer_First := Buffer'Last + 1;
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					      end Read_Buffer;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Translate is new Generic_Translate
 | 
				
			||||||
 | 
					                                   (Data_In  => Read_Buffer,
 | 
				
			||||||
 | 
					                                    Data_Out => Further);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Random_Elements.Reset (Gen);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Buffer := (others => 20);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Main : loop
 | 
				
			||||||
 | 
					         for J in Buffer'Range loop
 | 
				
			||||||
 | 
					            Buffer (J) := Random_Elements.Random (Gen);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Deflate_Init (Deflate);
 | 
				
			||||||
 | 
					            Inflate_Init (Inflate);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Buffer_First  := Buffer'First;
 | 
				
			||||||
 | 
					            Compare_First := Buffer'First;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Translate (Deflate);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if Compare_First /= Buffer'Last + 1 then
 | 
				
			||||||
 | 
					               raise Program_Error;
 | 
				
			||||||
 | 
					            end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Ada.Text_IO.Put_Line
 | 
				
			||||||
 | 
					              (Ada.Task_Identification.Image
 | 
				
			||||||
 | 
					                 (Ada.Task_Identification.Current_Task)
 | 
				
			||||||
 | 
					               & Stream_Element_Offset'Image (J)
 | 
				
			||||||
 | 
					               & ZLib.Count'Image (Total_Out (Deflate)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Close (Deflate);
 | 
				
			||||||
 | 
					            Close (Inflate);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            exit Main when Stop;
 | 
				
			||||||
 | 
					         end loop;
 | 
				
			||||||
 | 
					      end loop Main;
 | 
				
			||||||
 | 
					   exception
 | 
				
			||||||
 | 
					      when E : others =>
 | 
				
			||||||
 | 
					         Ada.Text_IO.Put_Line (Ada.Exceptions.Exception_Information (E));
 | 
				
			||||||
 | 
					         Stop := True;
 | 
				
			||||||
 | 
					   end Test_Task;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Test : array (1 .. 4) of Test_Task;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   pragma Unreferenced (Test);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Dummy : Character;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					   Ada.Text_IO.Get_Immediate (Dummy);
 | 
				
			||||||
 | 
					   Stop := True;
 | 
				
			||||||
 | 
					end MTest;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,156 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2003 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: read.adb,v 1.8 2004/05/31 10:53:40 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  Test/demo program for the generic read interface.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Ada.Numerics.Discrete_Random;
 | 
				
			||||||
 | 
					with Ada.Streams;
 | 
				
			||||||
 | 
					with Ada.Text_IO;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with ZLib;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure Read is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   use Ada.Streams;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------------------------
 | 
				
			||||||
 | 
					   --  Test configuration parameters --
 | 
				
			||||||
 | 
					   ------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   File_Size   : Stream_Element_Offset := 100_000;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Continuous  : constant Boolean          := False;
 | 
				
			||||||
 | 
					   --  If this constant is True, the test would be repeated again and again,
 | 
				
			||||||
 | 
					   --  with increment File_Size for every iteration.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Header      : constant ZLib.Header_Type := ZLib.Default;
 | 
				
			||||||
 | 
					   --  Do not use Header other than Default in ZLib versions 1.1.4 and older.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Init_Random : constant := 8;
 | 
				
			||||||
 | 
					   --  We are using the same random sequence, in case of we catch bug,
 | 
				
			||||||
 | 
					   --  so we would be able to reproduce it.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -- End --
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Pack_Size : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   Offset    : Stream_Element_Offset;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Filter     : ZLib.Filter_Type;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   subtype Visible_Symbols
 | 
				
			||||||
 | 
					      is Stream_Element range 16#20# .. 16#7E#;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   package Random_Elements is new
 | 
				
			||||||
 | 
					      Ada.Numerics.Discrete_Random (Visible_Symbols);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Gen : Random_Elements.Generator;
 | 
				
			||||||
 | 
					   Period  : constant Stream_Element_Offset := 200;
 | 
				
			||||||
 | 
					   --  Period constant variable for random generator not to be very random.
 | 
				
			||||||
 | 
					   --  Bigger period, harder random.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Read_Buffer : Stream_Element_Array (1 .. 2048);
 | 
				
			||||||
 | 
					   Read_First  : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   Read_Last   : Stream_Element_Offset;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Reset;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Read
 | 
				
			||||||
 | 
					     (Item : out Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last : out Stream_Element_Offset);
 | 
				
			||||||
 | 
					   --  this procedure is for generic instantiation of
 | 
				
			||||||
 | 
					   --  ZLib.Read
 | 
				
			||||||
 | 
					   --  reading data from the File_In.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Read is new ZLib.Read
 | 
				
			||||||
 | 
					                           (Read,
 | 
				
			||||||
 | 
					                            Read_Buffer,
 | 
				
			||||||
 | 
					                            Rest_First => Read_First,
 | 
				
			||||||
 | 
					                            Rest_Last  => Read_Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ----------
 | 
				
			||||||
 | 
					   -- Read --
 | 
				
			||||||
 | 
					   ----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Read
 | 
				
			||||||
 | 
					     (Item : out Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last : out Stream_Element_Offset) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Last := Stream_Element_Offset'Min
 | 
				
			||||||
 | 
					               (Item'Last,
 | 
				
			||||||
 | 
					                Item'First + File_Size - Offset);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      for J in Item'First .. Last loop
 | 
				
			||||||
 | 
					         if J < Item'First + Period then
 | 
				
			||||||
 | 
					            Item (J) := Random_Elements.Random (Gen);
 | 
				
			||||||
 | 
					         else
 | 
				
			||||||
 | 
					            Item (J) := Item (J - Period);
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Offset   := Offset + 1;
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					   end Read;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Reset --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Reset is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Random_Elements.Reset (Gen, Init_Random);
 | 
				
			||||||
 | 
					      Pack_Size := 0;
 | 
				
			||||||
 | 
					      Offset := 1;
 | 
				
			||||||
 | 
					      Read_First := Read_Buffer'Last + 1;
 | 
				
			||||||
 | 
					      Read_Last  := Read_Buffer'Last;
 | 
				
			||||||
 | 
					   end Reset;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					   Ada.Text_IO.Put_Line ("ZLib " & ZLib.Version);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   loop
 | 
				
			||||||
 | 
					      for Level in ZLib.Compression_Level'Range loop
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Ada.Text_IO.Put ("Level ="
 | 
				
			||||||
 | 
					            & ZLib.Compression_Level'Image (Level));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Deflate using generic instantiation.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Deflate_Init
 | 
				
			||||||
 | 
					               (Filter,
 | 
				
			||||||
 | 
					                Level,
 | 
				
			||||||
 | 
					                Header => Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Reset;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Ada.Text_IO.Put
 | 
				
			||||||
 | 
					           (Stream_Element_Offset'Image (File_Size) & " ->");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         loop
 | 
				
			||||||
 | 
					            declare
 | 
				
			||||||
 | 
					               Buffer : Stream_Element_Array (1 .. 1024);
 | 
				
			||||||
 | 
					               Last   : Stream_Element_Offset;
 | 
				
			||||||
 | 
					            begin
 | 
				
			||||||
 | 
					               Read (Filter, Buffer, Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					               Pack_Size := Pack_Size + Last - Buffer'First + 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					               exit when Last < Buffer'Last;
 | 
				
			||||||
 | 
					            end;
 | 
				
			||||||
 | 
					         end loop;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Ada.Text_IO.Put_Line (Stream_Element_Offset'Image (Pack_Size));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Close (Filter);
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      exit when not Continuous;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      File_Size := File_Size + 1;
 | 
				
			||||||
 | 
					   end loop;
 | 
				
			||||||
 | 
					end Read;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,65 @@
 | 
				
			||||||
 | 
					                        ZLib for Ada thick binding (ZLib.Ada)
 | 
				
			||||||
 | 
					                        Release 1.3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ZLib.Ada is a thick binding interface to the popular ZLib data
 | 
				
			||||||
 | 
					compression library, available at http://www.gzip.org/zlib/.
 | 
				
			||||||
 | 
					It provides Ada-style access to the ZLib C library.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        Here are the main changes since ZLib.Ada 1.2:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Attention: ZLib.Read generic routine have a initialization requirement
 | 
				
			||||||
 | 
					  for Read_Last parameter now. It is a bit incompatible with previous version,
 | 
				
			||||||
 | 
					  but extends functionality, we could use new parameters Allow_Read_Some and
 | 
				
			||||||
 | 
					  Flush now.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Added Is_Open routines to ZLib and ZLib.Streams packages.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Add pragma Assert to check Stream_Element is 8 bit.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Fix extraction to buffer with exact known decompressed size. Error reported by
 | 
				
			||||||
 | 
					  Steve Sangwine.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Fix definition of ULong (changed to unsigned_long), fix regression on 64 bits
 | 
				
			||||||
 | 
					  computers. Patch provided by Pascal Obry.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Add Status_Error exception definition.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Add pragma Assertion that Ada.Streams.Stream_Element size is 8 bit.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        How to build ZLib.Ada under GNAT
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					You should have the ZLib library already build on your computer, before
 | 
				
			||||||
 | 
					building ZLib.Ada. Make the directory of ZLib.Ada sources current and
 | 
				
			||||||
 | 
					issue the command:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  gnatmake test -largs -L<directory where libz.a is> -lz
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Or use the GNAT project file build for GNAT 3.15 or later:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  gnatmake -Pzlib.gpr -L<directory where libz.a is>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        How to build ZLib.Ada under Aonix ObjectAda for Win32 7.2.2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					1. Make a project with all *.ads and *.adb files from the distribution.
 | 
				
			||||||
 | 
					2. Build the libz.a library from the ZLib C sources.
 | 
				
			||||||
 | 
					3. Rename libz.a to z.lib.
 | 
				
			||||||
 | 
					4. Add the library z.lib to the project.
 | 
				
			||||||
 | 
					5. Add the libc.lib library from the ObjectAda distribution to the project.
 | 
				
			||||||
 | 
					6. Build the executable using test.adb as a main procedure.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        How to use ZLib.Ada
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The source files test.adb and read.adb are small demo programs that show
 | 
				
			||||||
 | 
					the main functionality of ZLib.Ada.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The routines from the package specifications are commented.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Homepage: http://zlib-ada.sourceforge.net/
 | 
				
			||||||
 | 
					Author: Dmitriy Anisimkov <anisimkov@yahoo.com>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Contributors: Pascal Obry <pascal@obry.org>, Steve Sangwine <sjs@essex.ac.uk>
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,463 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2003 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: test.adb,v 1.17 2003/08/12 12:13:30 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  The program has a few aims.
 | 
				
			||||||
 | 
					--  1. Test ZLib.Ada95 thick binding functionality.
 | 
				
			||||||
 | 
					--  2. Show the example of use main functionality of the ZLib.Ada95 binding.
 | 
				
			||||||
 | 
					--  3. Build this program automatically compile all ZLib.Ada95 packages under
 | 
				
			||||||
 | 
					--     GNAT Ada95 compiler.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with ZLib.Streams;
 | 
				
			||||||
 | 
					with Ada.Streams.Stream_IO;
 | 
				
			||||||
 | 
					with Ada.Numerics.Discrete_Random;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Ada.Text_IO;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Ada.Calendar;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure Test is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   use Ada.Streams;
 | 
				
			||||||
 | 
					   use Stream_IO;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------------------------
 | 
				
			||||||
 | 
					   --  Test configuration parameters --
 | 
				
			||||||
 | 
					   ------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   File_Size   : Count   := 100_000;
 | 
				
			||||||
 | 
					   Continuous  : constant Boolean := False;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Header      : constant ZLib.Header_Type := ZLib.Default;
 | 
				
			||||||
 | 
					                                              --  ZLib.None;
 | 
				
			||||||
 | 
					                                              --  ZLib.Auto;
 | 
				
			||||||
 | 
					                                              --  ZLib.GZip;
 | 
				
			||||||
 | 
					   --  Do not use Header other then Default in ZLib versions 1.1.4
 | 
				
			||||||
 | 
					   --  and older.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Strategy    : constant ZLib.Strategy_Type := ZLib.Default_Strategy;
 | 
				
			||||||
 | 
					   Init_Random : constant := 10;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -- End --
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   In_File_Name  : constant String := "testzlib.in";
 | 
				
			||||||
 | 
					   --  Name of the input file
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Z_File_Name   : constant String := "testzlib.zlb";
 | 
				
			||||||
 | 
					   --  Name of the compressed file.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Out_File_Name : constant String := "testzlib.out";
 | 
				
			||||||
 | 
					   --  Name of the decompressed file.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   File_In   : File_Type;
 | 
				
			||||||
 | 
					   File_Out  : File_Type;
 | 
				
			||||||
 | 
					   File_Back : File_Type;
 | 
				
			||||||
 | 
					   File_Z    : ZLib.Streams.Stream_Type;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Filter : ZLib.Filter_Type;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Time_Stamp : Ada.Calendar.Time;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Generate_File;
 | 
				
			||||||
 | 
					   --  Generate file of specified size with some random data.
 | 
				
			||||||
 | 
					   --  The random data is repeatable, for the good compression.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Compare_Streams
 | 
				
			||||||
 | 
					     (Left, Right : in out Root_Stream_Type'Class);
 | 
				
			||||||
 | 
					   --  The procedure comparing data in 2 streams.
 | 
				
			||||||
 | 
					   --  It is for compare data before and after compression/decompression.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Compare_Files (Left, Right : String);
 | 
				
			||||||
 | 
					   --  Compare files. Based on the Compare_Streams.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Copy_Streams
 | 
				
			||||||
 | 
					     (Source, Target : in out Root_Stream_Type'Class;
 | 
				
			||||||
 | 
					      Buffer_Size    : in     Stream_Element_Offset := 1024);
 | 
				
			||||||
 | 
					   --  Copying data from one stream to another. It is for test stream
 | 
				
			||||||
 | 
					   --  interface of the library.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Data_In
 | 
				
			||||||
 | 
					     (Item : out Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last : out Stream_Element_Offset);
 | 
				
			||||||
 | 
					   --  this procedure is for generic instantiation of
 | 
				
			||||||
 | 
					   --  ZLib.Generic_Translate.
 | 
				
			||||||
 | 
					   --  reading data from the File_In.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Data_Out (Item : in Stream_Element_Array);
 | 
				
			||||||
 | 
					   --  this procedure is for generic instantiation of
 | 
				
			||||||
 | 
					   --  ZLib.Generic_Translate.
 | 
				
			||||||
 | 
					   --  writing data to the File_Out.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Stamp;
 | 
				
			||||||
 | 
					   --  Store the timestamp to the local variable.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Print_Statistic (Msg : String; Data_Size : ZLib.Count);
 | 
				
			||||||
 | 
					   --  Print the time statistic with the message.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Translate is new ZLib.Generic_Translate
 | 
				
			||||||
 | 
					                                (Data_In  => Data_In,
 | 
				
			||||||
 | 
					                                 Data_Out => Data_Out);
 | 
				
			||||||
 | 
					   --  This procedure is moving data from File_In to File_Out
 | 
				
			||||||
 | 
					   --  with compression or decompression, depend on initialization of
 | 
				
			||||||
 | 
					   --  Filter parameter.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------------
 | 
				
			||||||
 | 
					   -- Compare_Files --
 | 
				
			||||||
 | 
					   -------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Compare_Files (Left, Right : String) is
 | 
				
			||||||
 | 
					      Left_File, Right_File : File_Type;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Open (Left_File, In_File, Left);
 | 
				
			||||||
 | 
					      Open (Right_File, In_File, Right);
 | 
				
			||||||
 | 
					      Compare_Streams (Stream (Left_File).all, Stream (Right_File).all);
 | 
				
			||||||
 | 
					      Close (Left_File);
 | 
				
			||||||
 | 
					      Close (Right_File);
 | 
				
			||||||
 | 
					   end Compare_Files;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------------
 | 
				
			||||||
 | 
					   -- Compare_Streams --
 | 
				
			||||||
 | 
					   ---------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Compare_Streams
 | 
				
			||||||
 | 
					     (Left, Right : in out Ada.Streams.Root_Stream_Type'Class)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      Left_Buffer, Right_Buffer : Stream_Element_Array (0 .. 16#FFF#);
 | 
				
			||||||
 | 
					      Left_Last, Right_Last : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      loop
 | 
				
			||||||
 | 
					         Read (Left, Left_Buffer, Left_Last);
 | 
				
			||||||
 | 
					         Read (Right, Right_Buffer, Right_Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         if Left_Last /= Right_Last then
 | 
				
			||||||
 | 
					            Ada.Text_IO.Put_Line ("Compare error :"
 | 
				
			||||||
 | 
					              & Stream_Element_Offset'Image (Left_Last)
 | 
				
			||||||
 | 
					              & " /= "
 | 
				
			||||||
 | 
					              & Stream_Element_Offset'Image (Right_Last));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            raise Constraint_Error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         elsif Left_Buffer (0 .. Left_Last)
 | 
				
			||||||
 | 
					               /= Right_Buffer (0 .. Right_Last)
 | 
				
			||||||
 | 
					         then
 | 
				
			||||||
 | 
					            Ada.Text_IO.Put_Line ("ERROR: IN and OUT files is not equal.");
 | 
				
			||||||
 | 
					            raise Constraint_Error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         exit when Left_Last < Left_Buffer'Last;
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					   end Compare_Streams;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					   -- Copy_Streams --
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Copy_Streams
 | 
				
			||||||
 | 
					     (Source, Target : in out Ada.Streams.Root_Stream_Type'Class;
 | 
				
			||||||
 | 
					      Buffer_Size    : in     Stream_Element_Offset := 1024)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      Buffer : Stream_Element_Array (1 .. Buffer_Size);
 | 
				
			||||||
 | 
					      Last   : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      loop
 | 
				
			||||||
 | 
					         Read  (Source, Buffer, Last);
 | 
				
			||||||
 | 
					         Write (Target, Buffer (1 .. Last));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         exit when Last < Buffer'Last;
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					   end Copy_Streams;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					   -- Data_In --
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Data_In
 | 
				
			||||||
 | 
					     (Item : out Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last : out Stream_Element_Offset) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Read (File_In, Item, Last);
 | 
				
			||||||
 | 
					   end Data_In;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					   -- Data_Out --
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Data_Out (Item : in Stream_Element_Array) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Write (File_Out, Item);
 | 
				
			||||||
 | 
					   end Data_Out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------------
 | 
				
			||||||
 | 
					   -- Generate_File --
 | 
				
			||||||
 | 
					   -------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Generate_File is
 | 
				
			||||||
 | 
					      subtype Visible_Symbols is Stream_Element range 16#20# .. 16#7E#;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      package Random_Elements is
 | 
				
			||||||
 | 
					         new Ada.Numerics.Discrete_Random (Visible_Symbols);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Gen    : Random_Elements.Generator;
 | 
				
			||||||
 | 
					      Buffer : Stream_Element_Array := (1 .. 77 => 16#20#) & 10;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Buffer_Count : constant Count := File_Size / Buffer'Length;
 | 
				
			||||||
 | 
					      --  Number of same buffers in the packet.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Density : constant Count := 30; --  from 0 to Buffer'Length - 2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Fill_Buffer (J, D : in Count);
 | 
				
			||||||
 | 
					      --  Change the part of the buffer.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      -----------------
 | 
				
			||||||
 | 
					      -- Fill_Buffer --
 | 
				
			||||||
 | 
					      -----------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Fill_Buffer (J, D : in Count) is
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         for K in 0 .. D loop
 | 
				
			||||||
 | 
					            Buffer
 | 
				
			||||||
 | 
					              (Stream_Element_Offset ((J + K) mod (Buffer'Length - 1) + 1))
 | 
				
			||||||
 | 
					             := Random_Elements.Random (Gen);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         end loop;
 | 
				
			||||||
 | 
					      end Fill_Buffer;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Random_Elements.Reset (Gen, Init_Random);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Create (File_In, Out_File, In_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Fill_Buffer (1, Buffer'Length - 2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      for J in 1 .. Buffer_Count loop
 | 
				
			||||||
 | 
					         Write (File_In, Buffer);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Fill_Buffer (J, Density);
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      --  fill remain size.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Write
 | 
				
			||||||
 | 
					        (File_In,
 | 
				
			||||||
 | 
					         Buffer
 | 
				
			||||||
 | 
					           (1 .. Stream_Element_Offset
 | 
				
			||||||
 | 
					                   (File_Size - Buffer'Length * Buffer_Count)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Flush (File_In);
 | 
				
			||||||
 | 
					      Close (File_In);
 | 
				
			||||||
 | 
					   end Generate_File;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------------
 | 
				
			||||||
 | 
					   -- Print_Statistic --
 | 
				
			||||||
 | 
					   ---------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Print_Statistic (Msg : String; Data_Size : ZLib.Count) is
 | 
				
			||||||
 | 
					      use Ada.Calendar;
 | 
				
			||||||
 | 
					      use Ada.Text_IO;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      package Count_IO is new Integer_IO (ZLib.Count);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Curr_Dur : Duration := Clock - Time_Stamp;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Put (Msg);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Set_Col (20);
 | 
				
			||||||
 | 
					      Ada.Text_IO.Put ("size =");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Count_IO.Put
 | 
				
			||||||
 | 
					        (Data_Size,
 | 
				
			||||||
 | 
					         Width => Stream_IO.Count'Image (File_Size)'Length);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Put_Line (" duration =" & Duration'Image (Curr_Dur));
 | 
				
			||||||
 | 
					   end Print_Statistic;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Stamp --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Stamp is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Time_Stamp := Ada.Calendar.Clock;
 | 
				
			||||||
 | 
					   end Stamp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					   Ada.Text_IO.Put_Line ("ZLib " & ZLib.Version);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   loop
 | 
				
			||||||
 | 
					      Generate_File;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      for Level in ZLib.Compression_Level'Range loop
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Ada.Text_IO.Put_Line ("Level ="
 | 
				
			||||||
 | 
					            & ZLib.Compression_Level'Image (Level));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Test generic interface.
 | 
				
			||||||
 | 
					         Open   (File_In, In_File, In_File_Name);
 | 
				
			||||||
 | 
					         Create (File_Out, Out_File, Z_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Stamp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Deflate using generic instantiation.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Deflate_Init
 | 
				
			||||||
 | 
					               (Filter   => Filter,
 | 
				
			||||||
 | 
					                Level    => Level,
 | 
				
			||||||
 | 
					                Strategy => Strategy,
 | 
				
			||||||
 | 
					                Header   => Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Translate (Filter);
 | 
				
			||||||
 | 
					         Print_Statistic ("Generic compress", ZLib.Total_Out (Filter));
 | 
				
			||||||
 | 
					         ZLib.Close (Filter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Close (File_In);
 | 
				
			||||||
 | 
					         Close (File_Out);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Open   (File_In, In_File, Z_File_Name);
 | 
				
			||||||
 | 
					         Create (File_Out, Out_File, Out_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Stamp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Inflate using generic instantiation.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Inflate_Init (Filter, Header => Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Translate (Filter);
 | 
				
			||||||
 | 
					         Print_Statistic ("Generic decompress", ZLib.Total_Out (Filter));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Close (Filter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Close (File_In);
 | 
				
			||||||
 | 
					         Close (File_Out);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Compare_Files (In_File_Name, Out_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Test stream interface.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Compress to the back stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Open   (File_In, In_File, In_File_Name);
 | 
				
			||||||
 | 
					         Create (File_Back, Out_File, Z_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Stamp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Create
 | 
				
			||||||
 | 
					           (Stream          => File_Z,
 | 
				
			||||||
 | 
					            Mode            => ZLib.Streams.Out_Stream,
 | 
				
			||||||
 | 
					            Back            => ZLib.Streams.Stream_Access
 | 
				
			||||||
 | 
					                                 (Stream (File_Back)),
 | 
				
			||||||
 | 
					            Back_Compressed => True,
 | 
				
			||||||
 | 
					            Level           => Level,
 | 
				
			||||||
 | 
					            Strategy        => Strategy,
 | 
				
			||||||
 | 
					            Header          => Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Copy_Streams
 | 
				
			||||||
 | 
					           (Source => Stream (File_In).all,
 | 
				
			||||||
 | 
					            Target => File_Z);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Flushing internal buffers to the back stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Flush (File_Z, ZLib.Finish);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Print_Statistic ("Write compress",
 | 
				
			||||||
 | 
					                          ZLib.Streams.Write_Total_Out (File_Z));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Close (File_Z);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Close (File_In);
 | 
				
			||||||
 | 
					         Close (File_Back);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Compare reading from original file and from
 | 
				
			||||||
 | 
					         --  decompression stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Open (File_In,   In_File, In_File_Name);
 | 
				
			||||||
 | 
					         Open (File_Back, In_File, Z_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Create
 | 
				
			||||||
 | 
					           (Stream          => File_Z,
 | 
				
			||||||
 | 
					            Mode            => ZLib.Streams.In_Stream,
 | 
				
			||||||
 | 
					            Back            => ZLib.Streams.Stream_Access
 | 
				
			||||||
 | 
					                                 (Stream (File_Back)),
 | 
				
			||||||
 | 
					            Back_Compressed => True,
 | 
				
			||||||
 | 
					            Header          => Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Stamp;
 | 
				
			||||||
 | 
					         Compare_Streams (Stream (File_In).all, File_Z);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Print_Statistic ("Read decompress",
 | 
				
			||||||
 | 
					                          ZLib.Streams.Read_Total_Out (File_Z));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Close (File_Z);
 | 
				
			||||||
 | 
					         Close (File_In);
 | 
				
			||||||
 | 
					         Close (File_Back);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Compress by reading from compression stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Open (File_Back, In_File, In_File_Name);
 | 
				
			||||||
 | 
					         Create (File_Out, Out_File, Z_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Create
 | 
				
			||||||
 | 
					           (Stream          => File_Z,
 | 
				
			||||||
 | 
					            Mode            => ZLib.Streams.In_Stream,
 | 
				
			||||||
 | 
					            Back            => ZLib.Streams.Stream_Access
 | 
				
			||||||
 | 
					                                 (Stream (File_Back)),
 | 
				
			||||||
 | 
					            Back_Compressed => False,
 | 
				
			||||||
 | 
					            Level           => Level,
 | 
				
			||||||
 | 
					            Strategy        => Strategy,
 | 
				
			||||||
 | 
					            Header          => Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Stamp;
 | 
				
			||||||
 | 
					         Copy_Streams
 | 
				
			||||||
 | 
					           (Source => File_Z,
 | 
				
			||||||
 | 
					            Target => Stream (File_Out).all);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Print_Statistic ("Read compress",
 | 
				
			||||||
 | 
					                          ZLib.Streams.Read_Total_Out (File_Z));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Close (File_Z);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Close (File_Out);
 | 
				
			||||||
 | 
					         Close (File_Back);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         --  Decompress to decompression stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Open   (File_In,   In_File, Z_File_Name);
 | 
				
			||||||
 | 
					         Create (File_Back, Out_File, Out_File_Name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Create
 | 
				
			||||||
 | 
					           (Stream          => File_Z,
 | 
				
			||||||
 | 
					            Mode            => ZLib.Streams.Out_Stream,
 | 
				
			||||||
 | 
					            Back            => ZLib.Streams.Stream_Access
 | 
				
			||||||
 | 
					                                 (Stream (File_Back)),
 | 
				
			||||||
 | 
					            Back_Compressed => False,
 | 
				
			||||||
 | 
					            Header          => Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Stamp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Copy_Streams
 | 
				
			||||||
 | 
					           (Source => Stream (File_In).all,
 | 
				
			||||||
 | 
					            Target => File_Z);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Print_Statistic ("Write decompress",
 | 
				
			||||||
 | 
					                          ZLib.Streams.Write_Total_Out (File_Z));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         ZLib.Streams.Close (File_Z);
 | 
				
			||||||
 | 
					         Close (File_In);
 | 
				
			||||||
 | 
					         Close (File_Back);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Compare_Files (In_File_Name, Out_File_Name);
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Ada.Text_IO.Put_Line (Count'Image (File_Size) & " Ok.");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      exit when not Continuous;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      File_Size := File_Size + 1;
 | 
				
			||||||
 | 
					   end loop;
 | 
				
			||||||
 | 
					end Test;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,225 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2003 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: zlib-streams.adb,v 1.10 2004/05/31 10:53:40 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Ada.Unchecked_Deallocation;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package body ZLib.Streams is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Close --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Close (Stream : in out Stream_Type) is
 | 
				
			||||||
 | 
					      procedure Free is new Ada.Unchecked_Deallocation
 | 
				
			||||||
 | 
					         (Stream_Element_Array, Buffer_Access);
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Stream.Mode = Out_Stream or Stream.Mode = Duplex then
 | 
				
			||||||
 | 
					         --  We should flush the data written by the writer.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Flush (Stream, Finish);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Close (Stream.Writer);
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Stream.Mode = In_Stream or Stream.Mode = Duplex then
 | 
				
			||||||
 | 
					         Close (Stream.Reader);
 | 
				
			||||||
 | 
					         Free (Stream.Buffer);
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Close;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------
 | 
				
			||||||
 | 
					   -- Create --
 | 
				
			||||||
 | 
					   ------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Create
 | 
				
			||||||
 | 
					     (Stream            :    out Stream_Type;
 | 
				
			||||||
 | 
					      Mode              : in     Stream_Mode;
 | 
				
			||||||
 | 
					      Back              : in     Stream_Access;
 | 
				
			||||||
 | 
					      Back_Compressed   : in     Boolean;
 | 
				
			||||||
 | 
					      Level             : in     Compression_Level := Default_Compression;
 | 
				
			||||||
 | 
					      Strategy          : in     Strategy_Type     := Default_Strategy;
 | 
				
			||||||
 | 
					      Header            : in     Header_Type       := Default;
 | 
				
			||||||
 | 
					      Read_Buffer_Size  : in     Ada.Streams.Stream_Element_Offset
 | 
				
			||||||
 | 
					                                    := Default_Buffer_Size;
 | 
				
			||||||
 | 
					      Write_Buffer_Size : in     Ada.Streams.Stream_Element_Offset
 | 
				
			||||||
 | 
					                                    := Default_Buffer_Size)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      subtype Buffer_Subtype is Stream_Element_Array (1 .. Read_Buffer_Size);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Init_Filter
 | 
				
			||||||
 | 
					         (Filter   : in out Filter_Type;
 | 
				
			||||||
 | 
					          Compress : in     Boolean);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      -----------------
 | 
				
			||||||
 | 
					      -- Init_Filter --
 | 
				
			||||||
 | 
					      -----------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Init_Filter
 | 
				
			||||||
 | 
					         (Filter   : in out Filter_Type;
 | 
				
			||||||
 | 
					          Compress : in     Boolean) is
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         if Compress then
 | 
				
			||||||
 | 
					            Deflate_Init
 | 
				
			||||||
 | 
					              (Filter, Level, Strategy, Header => Header);
 | 
				
			||||||
 | 
					         else
 | 
				
			||||||
 | 
					            Inflate_Init (Filter, Header => Header);
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					      end Init_Filter;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Stream.Back := Back;
 | 
				
			||||||
 | 
					      Stream.Mode := Mode;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Mode = Out_Stream or Mode = Duplex then
 | 
				
			||||||
 | 
					         Init_Filter (Stream.Writer, Back_Compressed);
 | 
				
			||||||
 | 
					         Stream.Buffer_Size := Write_Buffer_Size;
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					         Stream.Buffer_Size := 0;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Mode = In_Stream or Mode = Duplex then
 | 
				
			||||||
 | 
					         Init_Filter (Stream.Reader, not Back_Compressed);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Stream.Buffer     := new Buffer_Subtype;
 | 
				
			||||||
 | 
					         Stream.Rest_First := Stream.Buffer'Last + 1;
 | 
				
			||||||
 | 
					         Stream.Rest_Last  := Stream.Buffer'Last;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Create;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Flush --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Flush
 | 
				
			||||||
 | 
					     (Stream : in out Stream_Type;
 | 
				
			||||||
 | 
					      Mode   : in     Flush_Mode := Sync_Flush)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      Buffer : Stream_Element_Array (1 .. Stream.Buffer_Size);
 | 
				
			||||||
 | 
					      Last   : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      loop
 | 
				
			||||||
 | 
					         Flush (Stream.Writer, Buffer, Last, Mode);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Ada.Streams.Write (Stream.Back.all, Buffer (1 .. Last));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         exit when Last < Buffer'Last;
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					   end Flush;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					   -- Is_Open --
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Is_Open (Stream : Stream_Type) return Boolean is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Is_Open (Stream.Reader) or else Is_Open (Stream.Writer);
 | 
				
			||||||
 | 
					   end Is_Open;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ----------
 | 
				
			||||||
 | 
					   -- Read --
 | 
				
			||||||
 | 
					   ----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Read
 | 
				
			||||||
 | 
					     (Stream : in out Stream_Type;
 | 
				
			||||||
 | 
					      Item   :    out Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last   :    out Stream_Element_Offset)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Read
 | 
				
			||||||
 | 
					        (Item : out Stream_Element_Array;
 | 
				
			||||||
 | 
					         Last : out Stream_Element_Offset);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      ----------
 | 
				
			||||||
 | 
					      -- Read --
 | 
				
			||||||
 | 
					      ----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Read
 | 
				
			||||||
 | 
					        (Item : out Stream_Element_Array;
 | 
				
			||||||
 | 
					         Last : out Stream_Element_Offset) is
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         Ada.Streams.Read (Stream.Back.all, Item, Last);
 | 
				
			||||||
 | 
					      end Read;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Read is new ZLib.Read
 | 
				
			||||||
 | 
					         (Read       => Read,
 | 
				
			||||||
 | 
					          Buffer     => Stream.Buffer.all,
 | 
				
			||||||
 | 
					          Rest_First => Stream.Rest_First,
 | 
				
			||||||
 | 
					          Rest_Last  => Stream.Rest_Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Read (Stream.Reader, Item, Last);
 | 
				
			||||||
 | 
					   end Read;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------------
 | 
				
			||||||
 | 
					   -- Read_Total_In --
 | 
				
			||||||
 | 
					   -------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Read_Total_In (Stream : in Stream_Type) return Count is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Total_In (Stream.Reader);
 | 
				
			||||||
 | 
					   end Read_Total_In;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					   -- Read_Total_Out --
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Read_Total_Out (Stream : in Stream_Type) return Count is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Total_Out (Stream.Reader);
 | 
				
			||||||
 | 
					   end Read_Total_Out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Write --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Write
 | 
				
			||||||
 | 
					     (Stream : in out Stream_Type;
 | 
				
			||||||
 | 
					      Item   : in     Stream_Element_Array)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Write (Item : in Stream_Element_Array);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      -----------
 | 
				
			||||||
 | 
					      -- Write --
 | 
				
			||||||
 | 
					      -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Write (Item : in Stream_Element_Array) is
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         Ada.Streams.Write (Stream.Back.all, Item);
 | 
				
			||||||
 | 
					      end Write;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Write is new ZLib.Write
 | 
				
			||||||
 | 
					         (Write       => Write,
 | 
				
			||||||
 | 
					          Buffer_Size => Stream.Buffer_Size);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Write (Stream.Writer, Item, No_Flush);
 | 
				
			||||||
 | 
					   end Write;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					   -- Write_Total_In --
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Write_Total_In (Stream : in Stream_Type) return Count is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Total_In (Stream.Writer);
 | 
				
			||||||
 | 
					   end Write_Total_In;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------------
 | 
				
			||||||
 | 
					   -- Write_Total_Out --
 | 
				
			||||||
 | 
					   ---------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Write_Total_Out (Stream : in Stream_Type) return Count is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Total_Out (Stream.Writer);
 | 
				
			||||||
 | 
					   end Write_Total_Out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end ZLib.Streams;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,114 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2003 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: zlib-streams.ads,v 1.12 2004/05/31 10:53:40 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package ZLib.Streams is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Stream_Mode is (In_Stream, Out_Stream, Duplex);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Stream_Access is access all Ada.Streams.Root_Stream_Type'Class;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Stream_Type is
 | 
				
			||||||
 | 
					      new Ada.Streams.Root_Stream_Type with private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Read
 | 
				
			||||||
 | 
					     (Stream : in out Stream_Type;
 | 
				
			||||||
 | 
					      Item   :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last   :    out Ada.Streams.Stream_Element_Offset);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Write
 | 
				
			||||||
 | 
					     (Stream : in out Stream_Type;
 | 
				
			||||||
 | 
					      Item   : in     Ada.Streams.Stream_Element_Array);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Flush
 | 
				
			||||||
 | 
					     (Stream : in out Stream_Type;
 | 
				
			||||||
 | 
					      Mode   : in     Flush_Mode := Sync_Flush);
 | 
				
			||||||
 | 
					   --  Flush the written data to the back stream,
 | 
				
			||||||
 | 
					   --  all data placed to the compressor is flushing to the Back stream.
 | 
				
			||||||
 | 
					   --  Should not be used until necessary, because it is decreasing
 | 
				
			||||||
 | 
					   --  compression.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Read_Total_In (Stream : in Stream_Type) return Count;
 | 
				
			||||||
 | 
					   pragma Inline (Read_Total_In);
 | 
				
			||||||
 | 
					   --  Return total number of bytes read from back stream so far.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Read_Total_Out (Stream : in Stream_Type) return Count;
 | 
				
			||||||
 | 
					   pragma Inline (Read_Total_Out);
 | 
				
			||||||
 | 
					   --  Return total number of bytes read so far.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Write_Total_In (Stream : in Stream_Type) return Count;
 | 
				
			||||||
 | 
					   pragma Inline (Write_Total_In);
 | 
				
			||||||
 | 
					   --  Return total number of bytes written so far.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Write_Total_Out (Stream : in Stream_Type) return Count;
 | 
				
			||||||
 | 
					   pragma Inline (Write_Total_Out);
 | 
				
			||||||
 | 
					   --  Return total number of bytes written to the back stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Create
 | 
				
			||||||
 | 
					     (Stream            :    out Stream_Type;
 | 
				
			||||||
 | 
					      Mode              : in     Stream_Mode;
 | 
				
			||||||
 | 
					      Back              : in     Stream_Access;
 | 
				
			||||||
 | 
					      Back_Compressed   : in     Boolean;
 | 
				
			||||||
 | 
					      Level             : in     Compression_Level := Default_Compression;
 | 
				
			||||||
 | 
					      Strategy          : in     Strategy_Type     := Default_Strategy;
 | 
				
			||||||
 | 
					      Header            : in     Header_Type       := Default;
 | 
				
			||||||
 | 
					      Read_Buffer_Size  : in     Ada.Streams.Stream_Element_Offset
 | 
				
			||||||
 | 
					                                    := Default_Buffer_Size;
 | 
				
			||||||
 | 
					      Write_Buffer_Size : in     Ada.Streams.Stream_Element_Offset
 | 
				
			||||||
 | 
					                                    := Default_Buffer_Size);
 | 
				
			||||||
 | 
					   --  Create the Compression/Decompression stream.
 | 
				
			||||||
 | 
					   --  If mode is In_Stream then Write operation is disabled.
 | 
				
			||||||
 | 
					   --  If mode is Out_Stream then Read operation is disabled.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  If Back_Compressed is true then
 | 
				
			||||||
 | 
					   --  Data written to the Stream is compressing to the Back stream
 | 
				
			||||||
 | 
					   --  and data read from the Stream is decompressed data from the Back stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  If Back_Compressed is false then
 | 
				
			||||||
 | 
					   --  Data written to the Stream is decompressing to the Back stream
 | 
				
			||||||
 | 
					   --  and data read from the Stream is compressed data from the Back stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  !!! When the Need_Header is False ZLib-Ada is using undocumented
 | 
				
			||||||
 | 
					   --  ZLib 1.1.4 functionality to do not create/wait for ZLib headers.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Is_Open (Stream : Stream_Type) return Boolean;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Close (Stream : in out Stream_Type);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   use Ada.Streams;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Buffer_Access is access all Stream_Element_Array;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Stream_Type
 | 
				
			||||||
 | 
					     is new Root_Stream_Type with
 | 
				
			||||||
 | 
					   record
 | 
				
			||||||
 | 
					      Mode       : Stream_Mode;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Buffer     : Buffer_Access;
 | 
				
			||||||
 | 
					      Rest_First : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Rest_Last  : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      --  Buffer for Read operation.
 | 
				
			||||||
 | 
					      --  We need to have this buffer in the record
 | 
				
			||||||
 | 
					      --  because not all read data from back stream
 | 
				
			||||||
 | 
					      --  could be processed during the read operation.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Buffer_Size : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      --  Buffer size for write operation.
 | 
				
			||||||
 | 
					      --  We do not need to have this buffer
 | 
				
			||||||
 | 
					      --  in the record because all data could be
 | 
				
			||||||
 | 
					      --  processed in the write operation.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Back       : Stream_Access;
 | 
				
			||||||
 | 
					      Reader     : Filter_Type;
 | 
				
			||||||
 | 
					      Writer     : Filter_Type;
 | 
				
			||||||
 | 
					   end record;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end ZLib.Streams;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,141 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2003 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: zlib-thin.adb,v 1.8 2003/12/14 18:27:31 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package body ZLib.Thin is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ZLIB_VERSION  : constant Chars_Ptr := zlibVersion;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Z_Stream_Size : constant Int := Z_Stream'Size / System.Storage_Unit;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					   -- Avail_In --
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Avail_In (Strm : in Z_Stream) return UInt is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Strm.Avail_In;
 | 
				
			||||||
 | 
					   end Avail_In;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					   -- Avail_Out --
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Avail_Out (Strm : in Z_Stream) return UInt is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Strm.Avail_Out;
 | 
				
			||||||
 | 
					   end Avail_Out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					   -- Deflate_Init --
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Deflate_Init
 | 
				
			||||||
 | 
					     (strm       : Z_Streamp;
 | 
				
			||||||
 | 
					      level      : Int;
 | 
				
			||||||
 | 
					      method     : Int;
 | 
				
			||||||
 | 
					      windowBits : Int;
 | 
				
			||||||
 | 
					      memLevel   : Int;
 | 
				
			||||||
 | 
					      strategy   : Int)
 | 
				
			||||||
 | 
					      return       Int is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return deflateInit2
 | 
				
			||||||
 | 
					               (strm,
 | 
				
			||||||
 | 
					                level,
 | 
				
			||||||
 | 
					                method,
 | 
				
			||||||
 | 
					                windowBits,
 | 
				
			||||||
 | 
					                memLevel,
 | 
				
			||||||
 | 
					                strategy,
 | 
				
			||||||
 | 
					                ZLIB_VERSION,
 | 
				
			||||||
 | 
					                Z_Stream_Size);
 | 
				
			||||||
 | 
					   end Deflate_Init;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					   -- Inflate_Init --
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Inflate_Init (strm : Z_Streamp; windowBits : Int) return Int is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return inflateInit2 (strm, windowBits, ZLIB_VERSION, Z_Stream_Size);
 | 
				
			||||||
 | 
					   end Inflate_Init;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------------
 | 
				
			||||||
 | 
					   -- Last_Error_Message --
 | 
				
			||||||
 | 
					   ------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Last_Error_Message (Strm : in Z_Stream) return String is
 | 
				
			||||||
 | 
					      use Interfaces.C.Strings;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Strm.msg = Null_Ptr then
 | 
				
			||||||
 | 
					         return "";
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					         return Value (Strm.msg);
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Last_Error_Message;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------
 | 
				
			||||||
 | 
					   -- Set_In --
 | 
				
			||||||
 | 
					   ------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Set_In
 | 
				
			||||||
 | 
					     (Strm   : in out Z_Stream;
 | 
				
			||||||
 | 
					      Buffer : in     Voidp;
 | 
				
			||||||
 | 
					      Size   : in     UInt) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Strm.Next_In  := Buffer;
 | 
				
			||||||
 | 
					      Strm.Avail_In := Size;
 | 
				
			||||||
 | 
					   end Set_In;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					   -- Set_Mem_Func --
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Set_Mem_Func
 | 
				
			||||||
 | 
					     (Strm   : in out Z_Stream;
 | 
				
			||||||
 | 
					      Opaque : in     Voidp;
 | 
				
			||||||
 | 
					      Alloc  : in     alloc_func;
 | 
				
			||||||
 | 
					      Free   : in     free_func) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Strm.opaque := Opaque;
 | 
				
			||||||
 | 
					      Strm.zalloc := Alloc;
 | 
				
			||||||
 | 
					      Strm.zfree  := Free;
 | 
				
			||||||
 | 
					   end Set_Mem_Func;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					   -- Set_Out --
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Set_Out
 | 
				
			||||||
 | 
					     (Strm   : in out Z_Stream;
 | 
				
			||||||
 | 
					      Buffer : in     Voidp;
 | 
				
			||||||
 | 
					      Size   : in     UInt) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Strm.Next_Out  := Buffer;
 | 
				
			||||||
 | 
					      Strm.Avail_Out := Size;
 | 
				
			||||||
 | 
					   end Set_Out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					   -- Total_In --
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_In (Strm : in Z_Stream) return ULong is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Strm.Total_In;
 | 
				
			||||||
 | 
					   end Total_In;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					   -- Total_Out --
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_Out (Strm : in Z_Stream) return ULong is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Strm.Total_Out;
 | 
				
			||||||
 | 
					   end Total_Out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end ZLib.Thin;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,450 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2003 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: zlib-thin.ads,v 1.11 2004/07/23 06:33:11 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Interfaces.C.Strings;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with System;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private package ZLib.Thin is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  From zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   MAX_MEM_LEVEL : constant := 9;         --  zconf.h:105
 | 
				
			||||||
 | 
					                                          --  zconf.h:105
 | 
				
			||||||
 | 
					   MAX_WBITS : constant := 15;      --  zconf.h:115
 | 
				
			||||||
 | 
					                                    --  32K LZ77 window
 | 
				
			||||||
 | 
					                                    --  zconf.h:115
 | 
				
			||||||
 | 
					   SEEK_SET : constant := 8#0000#;  --  zconf.h:244
 | 
				
			||||||
 | 
					                                    --  Seek from beginning of file.
 | 
				
			||||||
 | 
					                                    --  zconf.h:244
 | 
				
			||||||
 | 
					   SEEK_CUR : constant := 1;        --  zconf.h:245
 | 
				
			||||||
 | 
					                                    --  Seek from current position.
 | 
				
			||||||
 | 
					                                    --  zconf.h:245
 | 
				
			||||||
 | 
					   SEEK_END : constant := 2;        --  zconf.h:246
 | 
				
			||||||
 | 
					                                    --  Set file pointer to EOF plus "offset"
 | 
				
			||||||
 | 
					                                    --  zconf.h:246
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Byte is new Interfaces.C.unsigned_char; --  8 bits
 | 
				
			||||||
 | 
					                                                --  zconf.h:214
 | 
				
			||||||
 | 
					   type UInt is new Interfaces.C.unsigned;      --  16 bits or more
 | 
				
			||||||
 | 
					                                                --  zconf.h:216
 | 
				
			||||||
 | 
					   type Int is new Interfaces.C.int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type ULong is new Interfaces.C.unsigned_long;     --  32 bits or more
 | 
				
			||||||
 | 
					                                                     --  zconf.h:217
 | 
				
			||||||
 | 
					   subtype Chars_Ptr is Interfaces.C.Strings.chars_ptr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type ULong_Access is access ULong;
 | 
				
			||||||
 | 
					   type Int_Access is access Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   subtype Voidp is System.Address;            --  zconf.h:232
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   subtype Byte_Access is Voidp;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Nul : constant Voidp := System.Null_Address;
 | 
				
			||||||
 | 
					   --  end from zconf
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Z_NO_FLUSH : constant := 8#0000#;   --  zlib.h:125
 | 
				
			||||||
 | 
					                                       --  zlib.h:125
 | 
				
			||||||
 | 
					   Z_PARTIAL_FLUSH : constant := 1;       --  zlib.h:126
 | 
				
			||||||
 | 
					                                          --  will be removed, use
 | 
				
			||||||
 | 
					                                          --  Z_SYNC_FLUSH instead
 | 
				
			||||||
 | 
					                                          --  zlib.h:126
 | 
				
			||||||
 | 
					   Z_SYNC_FLUSH : constant := 2;       --  zlib.h:127
 | 
				
			||||||
 | 
					                                       --  zlib.h:127
 | 
				
			||||||
 | 
					   Z_FULL_FLUSH : constant := 3;       --  zlib.h:128
 | 
				
			||||||
 | 
					                                       --  zlib.h:128
 | 
				
			||||||
 | 
					   Z_FINISH : constant := 4;        --  zlib.h:129
 | 
				
			||||||
 | 
					                                    --  zlib.h:129
 | 
				
			||||||
 | 
					   Z_OK : constant := 8#0000#;   --  zlib.h:132
 | 
				
			||||||
 | 
					                                 --  zlib.h:132
 | 
				
			||||||
 | 
					   Z_STREAM_END : constant := 1;       --  zlib.h:133
 | 
				
			||||||
 | 
					                                       --  zlib.h:133
 | 
				
			||||||
 | 
					   Z_NEED_DICT : constant := 2;        --  zlib.h:134
 | 
				
			||||||
 | 
					                                       --  zlib.h:134
 | 
				
			||||||
 | 
					   Z_ERRNO : constant := -1;        --  zlib.h:135
 | 
				
			||||||
 | 
					                                    --  zlib.h:135
 | 
				
			||||||
 | 
					   Z_STREAM_ERROR : constant := -2;       --  zlib.h:136
 | 
				
			||||||
 | 
					                                          --  zlib.h:136
 | 
				
			||||||
 | 
					   Z_DATA_ERROR : constant := -3;      --  zlib.h:137
 | 
				
			||||||
 | 
					                                       --  zlib.h:137
 | 
				
			||||||
 | 
					   Z_MEM_ERROR : constant := -4;       --  zlib.h:138
 | 
				
			||||||
 | 
					                                       --  zlib.h:138
 | 
				
			||||||
 | 
					   Z_BUF_ERROR : constant := -5;       --  zlib.h:139
 | 
				
			||||||
 | 
					                                       --  zlib.h:139
 | 
				
			||||||
 | 
					   Z_VERSION_ERROR : constant := -6;      --  zlib.h:140
 | 
				
			||||||
 | 
					                                          --  zlib.h:140
 | 
				
			||||||
 | 
					   Z_NO_COMPRESSION : constant := 8#0000#;   --  zlib.h:145
 | 
				
			||||||
 | 
					                                             --  zlib.h:145
 | 
				
			||||||
 | 
					   Z_BEST_SPEED : constant := 1;       --  zlib.h:146
 | 
				
			||||||
 | 
					                                       --  zlib.h:146
 | 
				
			||||||
 | 
					   Z_BEST_COMPRESSION : constant := 9;       --  zlib.h:147
 | 
				
			||||||
 | 
					                                             --  zlib.h:147
 | 
				
			||||||
 | 
					   Z_DEFAULT_COMPRESSION : constant := -1;      --  zlib.h:148
 | 
				
			||||||
 | 
					                                                --  zlib.h:148
 | 
				
			||||||
 | 
					   Z_FILTERED : constant := 1;      --  zlib.h:151
 | 
				
			||||||
 | 
					                                    --  zlib.h:151
 | 
				
			||||||
 | 
					   Z_HUFFMAN_ONLY : constant := 2;        --  zlib.h:152
 | 
				
			||||||
 | 
					                                          --  zlib.h:152
 | 
				
			||||||
 | 
					   Z_DEFAULT_STRATEGY : constant := 8#0000#; --  zlib.h:153
 | 
				
			||||||
 | 
					                                             --  zlib.h:153
 | 
				
			||||||
 | 
					   Z_BINARY : constant := 8#0000#;  --  zlib.h:156
 | 
				
			||||||
 | 
					                                    --  zlib.h:156
 | 
				
			||||||
 | 
					   Z_ASCII : constant := 1;      --  zlib.h:157
 | 
				
			||||||
 | 
					                                 --  zlib.h:157
 | 
				
			||||||
 | 
					   Z_UNKNOWN : constant := 2;       --  zlib.h:158
 | 
				
			||||||
 | 
					                                    --  zlib.h:158
 | 
				
			||||||
 | 
					   Z_DEFLATED : constant := 8;      --  zlib.h:161
 | 
				
			||||||
 | 
					                                    --  zlib.h:161
 | 
				
			||||||
 | 
					   Z_NULL : constant := 8#0000#; --  zlib.h:164
 | 
				
			||||||
 | 
					                                 --  for initializing zalloc, zfree, opaque
 | 
				
			||||||
 | 
					                                 --  zlib.h:164
 | 
				
			||||||
 | 
					   type gzFile is new Voidp;                  --  zlib.h:646
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Z_Stream is private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Z_Streamp is access all Z_Stream;     --  zlib.h:89
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type alloc_func is access function
 | 
				
			||||||
 | 
					     (Opaque : Voidp;
 | 
				
			||||||
 | 
					      Items  : UInt;
 | 
				
			||||||
 | 
					      Size   : UInt)
 | 
				
			||||||
 | 
					      return Voidp; --  zlib.h:63
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type free_func is access procedure (opaque : Voidp; address : Voidp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function zlibVersion return Chars_Ptr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Deflate (strm : Z_Streamp; flush : Int) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function DeflateEnd (strm : Z_Streamp) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Inflate (strm : Z_Streamp; flush : Int) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function InflateEnd (strm : Z_Streamp) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function deflateSetDictionary
 | 
				
			||||||
 | 
					     (strm       : Z_Streamp;
 | 
				
			||||||
 | 
					      dictionary : Byte_Access;
 | 
				
			||||||
 | 
					      dictLength : UInt)
 | 
				
			||||||
 | 
					      return       Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function deflateCopy (dest : Z_Streamp; source : Z_Streamp) return Int;
 | 
				
			||||||
 | 
					   --  zlib.h:478
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function deflateReset (strm : Z_Streamp) return Int; -- zlib.h:495
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function deflateParams
 | 
				
			||||||
 | 
					     (strm     : Z_Streamp;
 | 
				
			||||||
 | 
					      level    : Int;
 | 
				
			||||||
 | 
					      strategy : Int)
 | 
				
			||||||
 | 
					      return     Int;       -- zlib.h:506
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateSetDictionary
 | 
				
			||||||
 | 
					     (strm       : Z_Streamp;
 | 
				
			||||||
 | 
					      dictionary : Byte_Access;
 | 
				
			||||||
 | 
					      dictLength : UInt)
 | 
				
			||||||
 | 
					      return       Int; --  zlib.h:548
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateSync (strm : Z_Streamp) return Int;  --  zlib.h:565
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateReset (strm : Z_Streamp) return Int; --  zlib.h:580
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function compress
 | 
				
			||||||
 | 
					     (dest      : Byte_Access;
 | 
				
			||||||
 | 
					      destLen   : ULong_Access;
 | 
				
			||||||
 | 
					      source    : Byte_Access;
 | 
				
			||||||
 | 
					      sourceLen : ULong)
 | 
				
			||||||
 | 
					      return      Int;           -- zlib.h:601
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function compress2
 | 
				
			||||||
 | 
					     (dest      : Byte_Access;
 | 
				
			||||||
 | 
					      destLen   : ULong_Access;
 | 
				
			||||||
 | 
					      source    : Byte_Access;
 | 
				
			||||||
 | 
					      sourceLen : ULong;
 | 
				
			||||||
 | 
					      level     : Int)
 | 
				
			||||||
 | 
					      return      Int;          -- zlib.h:615
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function uncompress
 | 
				
			||||||
 | 
					     (dest      : Byte_Access;
 | 
				
			||||||
 | 
					      destLen   : ULong_Access;
 | 
				
			||||||
 | 
					      source    : Byte_Access;
 | 
				
			||||||
 | 
					      sourceLen : ULong)
 | 
				
			||||||
 | 
					      return      Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzopen (path : Chars_Ptr; mode : Chars_Ptr) return gzFile;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzdopen (fd : Int; mode : Chars_Ptr) return gzFile;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzsetparams
 | 
				
			||||||
 | 
					     (file     : gzFile;
 | 
				
			||||||
 | 
					      level    : Int;
 | 
				
			||||||
 | 
					      strategy : Int)
 | 
				
			||||||
 | 
					      return     Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzread
 | 
				
			||||||
 | 
					     (file : gzFile;
 | 
				
			||||||
 | 
					      buf  : Voidp;
 | 
				
			||||||
 | 
					      len  : UInt)
 | 
				
			||||||
 | 
					      return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzwrite
 | 
				
			||||||
 | 
					     (file : in gzFile;
 | 
				
			||||||
 | 
					      buf  : in Voidp;
 | 
				
			||||||
 | 
					      len  : in UInt)
 | 
				
			||||||
 | 
					      return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzprintf (file : in gzFile; format : in Chars_Ptr) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzputs (file : in gzFile; s : in Chars_Ptr) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzgets
 | 
				
			||||||
 | 
					     (file : gzFile;
 | 
				
			||||||
 | 
					      buf  : Chars_Ptr;
 | 
				
			||||||
 | 
					      len  : Int)
 | 
				
			||||||
 | 
					      return Chars_Ptr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzputc (file : gzFile; char : Int) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzgetc (file : gzFile) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzflush (file : gzFile; flush : Int) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzseek
 | 
				
			||||||
 | 
					     (file   : gzFile;
 | 
				
			||||||
 | 
					      offset : Int;
 | 
				
			||||||
 | 
					      whence : Int)
 | 
				
			||||||
 | 
					      return   Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzrewind (file : gzFile) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gztell (file : gzFile) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzeof (file : gzFile) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzclose (file : gzFile) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzerror (file : gzFile; errnum : Int_Access) return Chars_Ptr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function adler32
 | 
				
			||||||
 | 
					     (adler : ULong;
 | 
				
			||||||
 | 
					      buf   : Byte_Access;
 | 
				
			||||||
 | 
					      len   : UInt)
 | 
				
			||||||
 | 
					      return  ULong;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function crc32
 | 
				
			||||||
 | 
					     (crc  : ULong;
 | 
				
			||||||
 | 
					      buf  : Byte_Access;
 | 
				
			||||||
 | 
					      len  : UInt)
 | 
				
			||||||
 | 
					      return ULong;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function deflateInit
 | 
				
			||||||
 | 
					     (strm        : Z_Streamp;
 | 
				
			||||||
 | 
					      level       : Int;
 | 
				
			||||||
 | 
					      version     : Chars_Ptr;
 | 
				
			||||||
 | 
					      stream_size : Int)
 | 
				
			||||||
 | 
					      return        Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function deflateInit2
 | 
				
			||||||
 | 
					     (strm        : Z_Streamp;
 | 
				
			||||||
 | 
					      level       : Int;
 | 
				
			||||||
 | 
					      method      : Int;
 | 
				
			||||||
 | 
					      windowBits  : Int;
 | 
				
			||||||
 | 
					      memLevel    : Int;
 | 
				
			||||||
 | 
					      strategy    : Int;
 | 
				
			||||||
 | 
					      version     : Chars_Ptr;
 | 
				
			||||||
 | 
					      stream_size : Int)
 | 
				
			||||||
 | 
					      return        Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Deflate_Init
 | 
				
			||||||
 | 
					     (strm       : Z_Streamp;
 | 
				
			||||||
 | 
					      level      : Int;
 | 
				
			||||||
 | 
					      method     : Int;
 | 
				
			||||||
 | 
					      windowBits : Int;
 | 
				
			||||||
 | 
					      memLevel   : Int;
 | 
				
			||||||
 | 
					      strategy   : Int)
 | 
				
			||||||
 | 
					      return       Int;
 | 
				
			||||||
 | 
					   pragma Inline (Deflate_Init);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateInit
 | 
				
			||||||
 | 
					     (strm        : Z_Streamp;
 | 
				
			||||||
 | 
					      version     : Chars_Ptr;
 | 
				
			||||||
 | 
					      stream_size : Int)
 | 
				
			||||||
 | 
					      return        Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateInit2
 | 
				
			||||||
 | 
					     (strm        : in Z_Streamp;
 | 
				
			||||||
 | 
					      windowBits  : in Int;
 | 
				
			||||||
 | 
					      version     : in Chars_Ptr;
 | 
				
			||||||
 | 
					      stream_size : in Int)
 | 
				
			||||||
 | 
					      return      Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateBackInit
 | 
				
			||||||
 | 
					     (strm        : in Z_Streamp;
 | 
				
			||||||
 | 
					      windowBits  : in Int;
 | 
				
			||||||
 | 
					      window      : in Byte_Access;
 | 
				
			||||||
 | 
					      version     : in Chars_Ptr;
 | 
				
			||||||
 | 
					      stream_size : in Int)
 | 
				
			||||||
 | 
					      return      Int;
 | 
				
			||||||
 | 
					   --  Size of window have to be 2**windowBits.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Inflate_Init (strm : Z_Streamp; windowBits : Int) return Int;
 | 
				
			||||||
 | 
					   pragma Inline (Inflate_Init);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function zError (err : Int) return Chars_Ptr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateSyncPoint (z : Z_Streamp) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function get_crc_table return ULong_Access;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  Interface to the available fields of the z_stream structure.
 | 
				
			||||||
 | 
					   --  The application must update next_in and avail_in when avail_in has
 | 
				
			||||||
 | 
					   --  dropped to zero. It must update next_out and avail_out when avail_out
 | 
				
			||||||
 | 
					   --  has dropped to zero. The application must initialize zalloc, zfree and
 | 
				
			||||||
 | 
					   --  opaque before calling the init function.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Set_In
 | 
				
			||||||
 | 
					     (Strm   : in out Z_Stream;
 | 
				
			||||||
 | 
					      Buffer : in Voidp;
 | 
				
			||||||
 | 
					      Size   : in UInt);
 | 
				
			||||||
 | 
					   pragma Inline (Set_In);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Set_Out
 | 
				
			||||||
 | 
					     (Strm   : in out Z_Stream;
 | 
				
			||||||
 | 
					      Buffer : in Voidp;
 | 
				
			||||||
 | 
					      Size   : in UInt);
 | 
				
			||||||
 | 
					   pragma Inline (Set_Out);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Set_Mem_Func
 | 
				
			||||||
 | 
					     (Strm   : in out Z_Stream;
 | 
				
			||||||
 | 
					      Opaque : in Voidp;
 | 
				
			||||||
 | 
					      Alloc  : in alloc_func;
 | 
				
			||||||
 | 
					      Free   : in free_func);
 | 
				
			||||||
 | 
					   pragma Inline (Set_Mem_Func);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Last_Error_Message (Strm : in Z_Stream) return String;
 | 
				
			||||||
 | 
					   pragma Inline (Last_Error_Message);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Avail_Out (Strm : in Z_Stream) return UInt;
 | 
				
			||||||
 | 
					   pragma Inline (Avail_Out);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Avail_In (Strm : in Z_Stream) return UInt;
 | 
				
			||||||
 | 
					   pragma Inline (Avail_In);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_In (Strm : in Z_Stream) return ULong;
 | 
				
			||||||
 | 
					   pragma Inline (Total_In);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_Out (Strm : in Z_Stream) return ULong;
 | 
				
			||||||
 | 
					   pragma Inline (Total_Out);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function inflateCopy
 | 
				
			||||||
 | 
					     (dest   : in Z_Streamp;
 | 
				
			||||||
 | 
					      Source : in Z_Streamp)
 | 
				
			||||||
 | 
					      return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function compressBound (Source_Len : in ULong) return ULong;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function deflateBound
 | 
				
			||||||
 | 
					     (Strm       : in Z_Streamp;
 | 
				
			||||||
 | 
					      Source_Len : in ULong)
 | 
				
			||||||
 | 
					      return     ULong;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function gzungetc (C : in Int; File : in  gzFile) return Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function zlibCompileFlags return ULong;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Z_Stream is record            -- zlib.h:68
 | 
				
			||||||
 | 
					      Next_In   : Voidp      := Nul;  -- next input byte
 | 
				
			||||||
 | 
					      Avail_In  : UInt       := 0;    -- number of bytes available at next_in
 | 
				
			||||||
 | 
					      Total_In  : ULong      := 0;    -- total nb of input bytes read so far
 | 
				
			||||||
 | 
					      Next_Out  : Voidp      := Nul;  -- next output byte should be put there
 | 
				
			||||||
 | 
					      Avail_Out : UInt       := 0;    -- remaining free space at next_out
 | 
				
			||||||
 | 
					      Total_Out : ULong      := 0;    -- total nb of bytes output so far
 | 
				
			||||||
 | 
					      msg       : Chars_Ptr;          -- last error message, NULL if no error
 | 
				
			||||||
 | 
					      state     : Voidp;              -- not visible by applications
 | 
				
			||||||
 | 
					      zalloc    : alloc_func := null; -- used to allocate the internal state
 | 
				
			||||||
 | 
					      zfree     : free_func  := null; -- used to free the internal state
 | 
				
			||||||
 | 
					      opaque    : Voidp;              -- private data object passed to
 | 
				
			||||||
 | 
					                                      --  zalloc and zfree
 | 
				
			||||||
 | 
					      data_type : Int;                -- best guess about the data type:
 | 
				
			||||||
 | 
					                                      --  ascii or binary
 | 
				
			||||||
 | 
					      adler     : ULong;              -- adler32 value of the uncompressed
 | 
				
			||||||
 | 
					                                      --  data
 | 
				
			||||||
 | 
					      reserved  : ULong;              -- reserved for future use
 | 
				
			||||||
 | 
					   end record;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   pragma Convention (C, Z_Stream);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   pragma Import (C, zlibVersion, "zlibVersion");
 | 
				
			||||||
 | 
					   pragma Import (C, Deflate, "deflate");
 | 
				
			||||||
 | 
					   pragma Import (C, DeflateEnd, "deflateEnd");
 | 
				
			||||||
 | 
					   pragma Import (C, Inflate, "inflate");
 | 
				
			||||||
 | 
					   pragma Import (C, InflateEnd, "inflateEnd");
 | 
				
			||||||
 | 
					   pragma Import (C, deflateSetDictionary, "deflateSetDictionary");
 | 
				
			||||||
 | 
					   pragma Import (C, deflateCopy, "deflateCopy");
 | 
				
			||||||
 | 
					   pragma Import (C, deflateReset, "deflateReset");
 | 
				
			||||||
 | 
					   pragma Import (C, deflateParams, "deflateParams");
 | 
				
			||||||
 | 
					   pragma Import (C, inflateSetDictionary, "inflateSetDictionary");
 | 
				
			||||||
 | 
					   pragma Import (C, inflateSync, "inflateSync");
 | 
				
			||||||
 | 
					   pragma Import (C, inflateReset, "inflateReset");
 | 
				
			||||||
 | 
					   pragma Import (C, compress, "compress");
 | 
				
			||||||
 | 
					   pragma Import (C, compress2, "compress2");
 | 
				
			||||||
 | 
					   pragma Import (C, uncompress, "uncompress");
 | 
				
			||||||
 | 
					   pragma Import (C, gzopen, "gzopen");
 | 
				
			||||||
 | 
					   pragma Import (C, gzdopen, "gzdopen");
 | 
				
			||||||
 | 
					   pragma Import (C, gzsetparams, "gzsetparams");
 | 
				
			||||||
 | 
					   pragma Import (C, gzread, "gzread");
 | 
				
			||||||
 | 
					   pragma Import (C, gzwrite, "gzwrite");
 | 
				
			||||||
 | 
					   pragma Import (C, gzprintf, "gzprintf");
 | 
				
			||||||
 | 
					   pragma Import (C, gzputs, "gzputs");
 | 
				
			||||||
 | 
					   pragma Import (C, gzgets, "gzgets");
 | 
				
			||||||
 | 
					   pragma Import (C, gzputc, "gzputc");
 | 
				
			||||||
 | 
					   pragma Import (C, gzgetc, "gzgetc");
 | 
				
			||||||
 | 
					   pragma Import (C, gzflush, "gzflush");
 | 
				
			||||||
 | 
					   pragma Import (C, gzseek, "gzseek");
 | 
				
			||||||
 | 
					   pragma Import (C, gzrewind, "gzrewind");
 | 
				
			||||||
 | 
					   pragma Import (C, gztell, "gztell");
 | 
				
			||||||
 | 
					   pragma Import (C, gzeof, "gzeof");
 | 
				
			||||||
 | 
					   pragma Import (C, gzclose, "gzclose");
 | 
				
			||||||
 | 
					   pragma Import (C, gzerror, "gzerror");
 | 
				
			||||||
 | 
					   pragma Import (C, adler32, "adler32");
 | 
				
			||||||
 | 
					   pragma Import (C, crc32, "crc32");
 | 
				
			||||||
 | 
					   pragma Import (C, deflateInit, "deflateInit_");
 | 
				
			||||||
 | 
					   pragma Import (C, inflateInit, "inflateInit_");
 | 
				
			||||||
 | 
					   pragma Import (C, deflateInit2, "deflateInit2_");
 | 
				
			||||||
 | 
					   pragma Import (C, inflateInit2, "inflateInit2_");
 | 
				
			||||||
 | 
					   pragma Import (C, zError, "zError");
 | 
				
			||||||
 | 
					   pragma Import (C, inflateSyncPoint, "inflateSyncPoint");
 | 
				
			||||||
 | 
					   pragma Import (C, get_crc_table, "get_crc_table");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  since zlib 1.2.0:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   pragma Import (C, inflateCopy, "inflateCopy");
 | 
				
			||||||
 | 
					   pragma Import (C, compressBound, "compressBound");
 | 
				
			||||||
 | 
					   pragma Import (C, deflateBound, "deflateBound");
 | 
				
			||||||
 | 
					   pragma Import (C, gzungetc, "gzungetc");
 | 
				
			||||||
 | 
					   pragma Import (C, zlibCompileFlags, "zlibCompileFlags");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   pragma Import (C, inflateBackInit, "inflateBackInit_");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  I stopped binding the inflateBack routines, because realize that
 | 
				
			||||||
 | 
					   --  it does not support zlib and gzip headers for now, and have no
 | 
				
			||||||
 | 
					   --  symmetric deflateBack routines.
 | 
				
			||||||
 | 
					   --  ZLib-Ada is symmetric regarding deflate/inflate data transformation
 | 
				
			||||||
 | 
					   --  and has a similar generic callback interface for the
 | 
				
			||||||
 | 
					   --  deflate/inflate transformation based on the regular Deflate/Inflate
 | 
				
			||||||
 | 
					   --  routines.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  pragma Import (C, inflateBack, "inflateBack");
 | 
				
			||||||
 | 
					   --  pragma Import (C, inflateBackEnd, "inflateBackEnd");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end ZLib.Thin;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,701 @@
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					--  ZLib for Ada thick binding.                               --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Copyright (C) 2002-2004 Dmitriy Anisimkov                 --
 | 
				
			||||||
 | 
					--                                                            --
 | 
				
			||||||
 | 
					--  Open source license information is in the zlib.ads file.  --
 | 
				
			||||||
 | 
					----------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: zlib.adb,v 1.31 2004/09/06 06:53:19 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Ada.Exceptions;
 | 
				
			||||||
 | 
					with Ada.Unchecked_Conversion;
 | 
				
			||||||
 | 
					with Ada.Unchecked_Deallocation;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Interfaces.C.Strings;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with ZLib.Thin;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package body ZLib is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   use type Thin.Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Z_Stream is new Thin.Z_Stream;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Return_Code_Enum is
 | 
				
			||||||
 | 
					      (OK,
 | 
				
			||||||
 | 
					       STREAM_END,
 | 
				
			||||||
 | 
					       NEED_DICT,
 | 
				
			||||||
 | 
					       ERRNO,
 | 
				
			||||||
 | 
					       STREAM_ERROR,
 | 
				
			||||||
 | 
					       DATA_ERROR,
 | 
				
			||||||
 | 
					       MEM_ERROR,
 | 
				
			||||||
 | 
					       BUF_ERROR,
 | 
				
			||||||
 | 
					       VERSION_ERROR);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Flate_Step_Function is access
 | 
				
			||||||
 | 
					     function (Strm : in Thin.Z_Streamp; Flush : in Thin.Int) return Thin.Int;
 | 
				
			||||||
 | 
					   pragma Convention (C, Flate_Step_Function);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Flate_End_Function is access
 | 
				
			||||||
 | 
					      function (Ctrm : in Thin.Z_Streamp) return Thin.Int;
 | 
				
			||||||
 | 
					   pragma Convention (C, Flate_End_Function);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Flate_Type is record
 | 
				
			||||||
 | 
					      Step : Flate_Step_Function;
 | 
				
			||||||
 | 
					      Done : Flate_End_Function;
 | 
				
			||||||
 | 
					   end record;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   subtype Footer_Array is Stream_Element_Array (1 .. 8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Simple_GZip_Header : constant Stream_Element_Array (1 .. 10)
 | 
				
			||||||
 | 
					     := (16#1f#, 16#8b#,                 --  Magic header
 | 
				
			||||||
 | 
					         16#08#,                         --  Z_DEFLATED
 | 
				
			||||||
 | 
					         16#00#,                         --  Flags
 | 
				
			||||||
 | 
					         16#00#, 16#00#, 16#00#, 16#00#, --  Time
 | 
				
			||||||
 | 
					         16#00#,                         --  XFlags
 | 
				
			||||||
 | 
					         16#03#                          --  OS code
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					   --  The simplest gzip header is not for informational, but just for
 | 
				
			||||||
 | 
					   --  gzip format compatibility.
 | 
				
			||||||
 | 
					   --  Note that some code below is using assumption
 | 
				
			||||||
 | 
					   --  Simple_GZip_Header'Last > Footer_Array'Last, so do not make
 | 
				
			||||||
 | 
					   --  Simple_GZip_Header'Last <= Footer_Array'Last.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Return_Code : constant array (Thin.Int range <>) of Return_Code_Enum
 | 
				
			||||||
 | 
					     := (0 => OK,
 | 
				
			||||||
 | 
					         1 => STREAM_END,
 | 
				
			||||||
 | 
					         2 => NEED_DICT,
 | 
				
			||||||
 | 
					        -1 => ERRNO,
 | 
				
			||||||
 | 
					        -2 => STREAM_ERROR,
 | 
				
			||||||
 | 
					        -3 => DATA_ERROR,
 | 
				
			||||||
 | 
					        -4 => MEM_ERROR,
 | 
				
			||||||
 | 
					        -5 => BUF_ERROR,
 | 
				
			||||||
 | 
					        -6 => VERSION_ERROR);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Flate : constant array (Boolean) of Flate_Type
 | 
				
			||||||
 | 
					     := (True  => (Step => Thin.Deflate'Access,
 | 
				
			||||||
 | 
					                   Done => Thin.DeflateEnd'Access),
 | 
				
			||||||
 | 
					         False => (Step => Thin.Inflate'Access,
 | 
				
			||||||
 | 
					                   Done => Thin.InflateEnd'Access));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Flush_Finish : constant array (Boolean) of Flush_Mode
 | 
				
			||||||
 | 
					     := (True => Finish, False => No_Flush);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Raise_Error (Stream : in Z_Stream);
 | 
				
			||||||
 | 
					   pragma Inline (Raise_Error);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Raise_Error (Message : in String);
 | 
				
			||||||
 | 
					   pragma Inline (Raise_Error);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Check_Error (Stream : in Z_Stream; Code : in Thin.Int);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Free is new Ada.Unchecked_Deallocation
 | 
				
			||||||
 | 
					      (Z_Stream, Z_Stream_Access);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function To_Thin_Access is new Ada.Unchecked_Conversion
 | 
				
			||||||
 | 
					     (Z_Stream_Access, Thin.Z_Streamp);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Translate_GZip
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      In_Data   : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      In_Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode);
 | 
				
			||||||
 | 
					   --  Separate translate routine for make gzip header.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Translate_Auto
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      In_Data   : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      In_Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode);
 | 
				
			||||||
 | 
					   --  translate routine without additional headers.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------------
 | 
				
			||||||
 | 
					   -- Check_Error --
 | 
				
			||||||
 | 
					   -----------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Check_Error (Stream : in Z_Stream; Code : in Thin.Int) is
 | 
				
			||||||
 | 
					      use type Thin.Int;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Code /= Thin.Z_OK then
 | 
				
			||||||
 | 
					         Raise_Error
 | 
				
			||||||
 | 
					            (Return_Code_Enum'Image (Return_Code (Code))
 | 
				
			||||||
 | 
					              & ": " & Last_Error_Message (Stream));
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Check_Error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Close --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Close
 | 
				
			||||||
 | 
					     (Filter       : in out Filter_Type;
 | 
				
			||||||
 | 
					      Ignore_Error : in     Boolean := False)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      Code : Thin.Int;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if not Ignore_Error and then not Is_Open (Filter) then
 | 
				
			||||||
 | 
					         raise Status_Error;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Code := Flate (Filter.Compression).Done (To_Thin_Access (Filter.Strm));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Ignore_Error or else Code = Thin.Z_OK then
 | 
				
			||||||
 | 
					         Free (Filter.Strm);
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					         declare
 | 
				
			||||||
 | 
					            Error_Message : constant String
 | 
				
			||||||
 | 
					              := Last_Error_Message (Filter.Strm.all);
 | 
				
			||||||
 | 
					         begin
 | 
				
			||||||
 | 
					            Free (Filter.Strm);
 | 
				
			||||||
 | 
					            Ada.Exceptions.Raise_Exception
 | 
				
			||||||
 | 
					               (ZLib_Error'Identity,
 | 
				
			||||||
 | 
					                Return_Code_Enum'Image (Return_Code (Code))
 | 
				
			||||||
 | 
					                  & ": " & Error_Message);
 | 
				
			||||||
 | 
					         end;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Close;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- CRC32 --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function CRC32
 | 
				
			||||||
 | 
					     (CRC  : in Unsigned_32;
 | 
				
			||||||
 | 
					      Data : in Ada.Streams.Stream_Element_Array)
 | 
				
			||||||
 | 
					      return Unsigned_32
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      use Thin;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Unsigned_32 (crc32 (ULong (CRC),
 | 
				
			||||||
 | 
					                                 Data'Address,
 | 
				
			||||||
 | 
					                                 Data'Length));
 | 
				
			||||||
 | 
					   end CRC32;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure CRC32
 | 
				
			||||||
 | 
					     (CRC  : in out Unsigned_32;
 | 
				
			||||||
 | 
					      Data : in     Ada.Streams.Stream_Element_Array) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      CRC := CRC32 (CRC, Data);
 | 
				
			||||||
 | 
					   end CRC32;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					   -- Deflate_Init --
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Deflate_Init
 | 
				
			||||||
 | 
					     (Filter       : in out Filter_Type;
 | 
				
			||||||
 | 
					      Level        : in     Compression_Level  := Default_Compression;
 | 
				
			||||||
 | 
					      Strategy     : in     Strategy_Type      := Default_Strategy;
 | 
				
			||||||
 | 
					      Method       : in     Compression_Method := Deflated;
 | 
				
			||||||
 | 
					      Window_Bits  : in     Window_Bits_Type   := Default_Window_Bits;
 | 
				
			||||||
 | 
					      Memory_Level : in     Memory_Level_Type  := Default_Memory_Level;
 | 
				
			||||||
 | 
					      Header       : in     Header_Type        := Default)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      use type Thin.Int;
 | 
				
			||||||
 | 
					      Win_Bits : Thin.Int := Thin.Int (Window_Bits);
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Is_Open (Filter) then
 | 
				
			||||||
 | 
					         raise Status_Error;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      --  We allow ZLib to make header only in case of default header type.
 | 
				
			||||||
 | 
					      --  Otherwise we would either do header by ourselves, or do not do
 | 
				
			||||||
 | 
					      --  header at all.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Header = None or else Header = GZip then
 | 
				
			||||||
 | 
					         Win_Bits := -Win_Bits;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      --  For the GZip CRC calculation and make headers.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Header = GZip then
 | 
				
			||||||
 | 
					         Filter.CRC    := 0;
 | 
				
			||||||
 | 
					         Filter.Offset := Simple_GZip_Header'First;
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					         Filter.Offset := Simple_GZip_Header'Last + 1;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Filter.Strm        := new Z_Stream;
 | 
				
			||||||
 | 
					      Filter.Compression := True;
 | 
				
			||||||
 | 
					      Filter.Stream_End  := False;
 | 
				
			||||||
 | 
					      Filter.Header      := Header;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Thin.Deflate_Init
 | 
				
			||||||
 | 
					           (To_Thin_Access (Filter.Strm),
 | 
				
			||||||
 | 
					            Level      => Thin.Int (Level),
 | 
				
			||||||
 | 
					            method     => Thin.Int (Method),
 | 
				
			||||||
 | 
					            windowBits => Win_Bits,
 | 
				
			||||||
 | 
					            memLevel   => Thin.Int (Memory_Level),
 | 
				
			||||||
 | 
					            strategy   => Thin.Int (Strategy)) /= Thin.Z_OK
 | 
				
			||||||
 | 
					      then
 | 
				
			||||||
 | 
					         Raise_Error (Filter.Strm.all);
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Deflate_Init;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Flush --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Flush
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      No_Data : Stream_Element_Array := (1 .. 0 => 0);
 | 
				
			||||||
 | 
					      Last    : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Translate (Filter, No_Data, Last, Out_Data, Out_Last, Flush);
 | 
				
			||||||
 | 
					   end Flush;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------------------
 | 
				
			||||||
 | 
					   -- Generic_Translate --
 | 
				
			||||||
 | 
					   -----------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Generic_Translate
 | 
				
			||||||
 | 
					     (Filter          : in out ZLib.Filter_Type;
 | 
				
			||||||
 | 
					      In_Buffer_Size  : in     Integer := Default_Buffer_Size;
 | 
				
			||||||
 | 
					      Out_Buffer_Size : in     Integer := Default_Buffer_Size)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      In_Buffer  : Stream_Element_Array
 | 
				
			||||||
 | 
					                     (1 .. Stream_Element_Offset (In_Buffer_Size));
 | 
				
			||||||
 | 
					      Out_Buffer : Stream_Element_Array
 | 
				
			||||||
 | 
					                     (1 .. Stream_Element_Offset (Out_Buffer_Size));
 | 
				
			||||||
 | 
					      Last       : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      In_Last    : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      In_First   : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Last   : Stream_Element_Offset;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Main : loop
 | 
				
			||||||
 | 
					         Data_In (In_Buffer, Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         In_First := In_Buffer'First;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         loop
 | 
				
			||||||
 | 
					            Translate
 | 
				
			||||||
 | 
					              (Filter   => Filter,
 | 
				
			||||||
 | 
					               In_Data  => In_Buffer (In_First .. Last),
 | 
				
			||||||
 | 
					               In_Last  => In_Last,
 | 
				
			||||||
 | 
					               Out_Data => Out_Buffer,
 | 
				
			||||||
 | 
					               Out_Last => Out_Last,
 | 
				
			||||||
 | 
					               Flush    => Flush_Finish (Last < In_Buffer'First));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if Out_Buffer'First <= Out_Last then
 | 
				
			||||||
 | 
					               Data_Out (Out_Buffer (Out_Buffer'First .. Out_Last));
 | 
				
			||||||
 | 
					            end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            exit Main when Stream_End (Filter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            --  The end of in buffer.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            exit when In_Last = Last;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            In_First := In_Last + 1;
 | 
				
			||||||
 | 
					         end loop;
 | 
				
			||||||
 | 
					      end loop Main;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   end Generic_Translate;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					   -- Inflate_Init --
 | 
				
			||||||
 | 
					   ------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Inflate_Init
 | 
				
			||||||
 | 
					     (Filter      : in out Filter_Type;
 | 
				
			||||||
 | 
					      Window_Bits : in     Window_Bits_Type := Default_Window_Bits;
 | 
				
			||||||
 | 
					      Header      : in     Header_Type      := Default)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      use type Thin.Int;
 | 
				
			||||||
 | 
					      Win_Bits : Thin.Int := Thin.Int (Window_Bits);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Check_Version;
 | 
				
			||||||
 | 
					      --  Check the latest header types compatibility.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Check_Version is
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         if Version <= "1.1.4" then
 | 
				
			||||||
 | 
					            Raise_Error
 | 
				
			||||||
 | 
					              ("Inflate header type " & Header_Type'Image (Header)
 | 
				
			||||||
 | 
					               & " incompatible with ZLib version " & Version);
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					      end Check_Version;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Is_Open (Filter) then
 | 
				
			||||||
 | 
					         raise Status_Error;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      case Header is
 | 
				
			||||||
 | 
					         when None =>
 | 
				
			||||||
 | 
					            Check_Version;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            --  Inflate data without headers determined
 | 
				
			||||||
 | 
					            --  by negative Win_Bits.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Win_Bits := -Win_Bits;
 | 
				
			||||||
 | 
					         when GZip =>
 | 
				
			||||||
 | 
					            Check_Version;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            --  Inflate gzip data defined by flag 16.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Win_Bits := Win_Bits + 16;
 | 
				
			||||||
 | 
					         when Auto =>
 | 
				
			||||||
 | 
					            Check_Version;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            --  Inflate with automatic detection
 | 
				
			||||||
 | 
					            --  of gzip or native header defined by flag 32.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Win_Bits := Win_Bits + 32;
 | 
				
			||||||
 | 
					         when Default => null;
 | 
				
			||||||
 | 
					      end case;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Filter.Strm        := new Z_Stream;
 | 
				
			||||||
 | 
					      Filter.Compression := False;
 | 
				
			||||||
 | 
					      Filter.Stream_End  := False;
 | 
				
			||||||
 | 
					      Filter.Header      := Header;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Thin.Inflate_Init
 | 
				
			||||||
 | 
					         (To_Thin_Access (Filter.Strm), Win_Bits) /= Thin.Z_OK
 | 
				
			||||||
 | 
					      then
 | 
				
			||||||
 | 
					         Raise_Error (Filter.Strm.all);
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Inflate_Init;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					   -- Is_Open --
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Is_Open (Filter : in Filter_Type) return Boolean is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Filter.Strm /= null;
 | 
				
			||||||
 | 
					   end Is_Open;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------------
 | 
				
			||||||
 | 
					   -- Raise_Error --
 | 
				
			||||||
 | 
					   -----------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Raise_Error (Message : in String) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Ada.Exceptions.Raise_Exception (ZLib_Error'Identity, Message);
 | 
				
			||||||
 | 
					   end Raise_Error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Raise_Error (Stream : in Z_Stream) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Raise_Error (Last_Error_Message (Stream));
 | 
				
			||||||
 | 
					   end Raise_Error;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ----------
 | 
				
			||||||
 | 
					   -- Read --
 | 
				
			||||||
 | 
					   ----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Read
 | 
				
			||||||
 | 
					     (Filter : in out Filter_Type;
 | 
				
			||||||
 | 
					      Item   :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush  : in     Flush_Mode := No_Flush)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      In_Last    : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Item_First : Ada.Streams.Stream_Element_Offset := Item'First;
 | 
				
			||||||
 | 
					      V_Flush    : Flush_Mode := Flush;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      pragma Assert (Rest_First in Buffer'First .. Buffer'Last + 1);
 | 
				
			||||||
 | 
					      pragma Assert (Rest_Last in Buffer'First - 1 .. Buffer'Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      loop
 | 
				
			||||||
 | 
					         if Rest_Last = Buffer'First - 1 then
 | 
				
			||||||
 | 
					            V_Flush := Finish;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         elsif Rest_First > Rest_Last then
 | 
				
			||||||
 | 
					            Read (Buffer, Rest_Last);
 | 
				
			||||||
 | 
					            Rest_First := Buffer'First;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if Rest_Last < Buffer'First then
 | 
				
			||||||
 | 
					               V_Flush := Finish;
 | 
				
			||||||
 | 
					            end if;
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Translate
 | 
				
			||||||
 | 
					           (Filter   => Filter,
 | 
				
			||||||
 | 
					            In_Data  => Buffer (Rest_First .. Rest_Last),
 | 
				
			||||||
 | 
					            In_Last  => In_Last,
 | 
				
			||||||
 | 
					            Out_Data => Item (Item_First .. Item'Last),
 | 
				
			||||||
 | 
					            Out_Last => Last,
 | 
				
			||||||
 | 
					            Flush    => V_Flush);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Rest_First := In_Last + 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         exit when Stream_End (Filter)
 | 
				
			||||||
 | 
					           or else Last = Item'Last
 | 
				
			||||||
 | 
					           or else (Last >= Item'First and then Allow_Read_Some);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Item_First := Last + 1;
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					   end Read;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ----------------
 | 
				
			||||||
 | 
					   -- Stream_End --
 | 
				
			||||||
 | 
					   ----------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Stream_End (Filter : in Filter_Type) return Boolean is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Filter.Header = GZip and Filter.Compression then
 | 
				
			||||||
 | 
					         return Filter.Stream_End
 | 
				
			||||||
 | 
					            and then Filter.Offset = Footer_Array'Last + 1;
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					         return Filter.Stream_End;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Stream_End;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					   -- Total_In --
 | 
				
			||||||
 | 
					   --------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_In (Filter : in Filter_Type) return Count is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Count (Thin.Total_In (To_Thin_Access (Filter.Strm).all));
 | 
				
			||||||
 | 
					   end Total_In;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					   -- Total_Out --
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_Out (Filter : in Filter_Type) return Count is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Count (Thin.Total_Out (To_Thin_Access (Filter.Strm).all));
 | 
				
			||||||
 | 
					   end Total_Out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					   -- Translate --
 | 
				
			||||||
 | 
					   ---------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Translate
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      In_Data   : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      In_Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode) is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Filter.Header = GZip and then Filter.Compression then
 | 
				
			||||||
 | 
					         Translate_GZip
 | 
				
			||||||
 | 
					           (Filter   => Filter,
 | 
				
			||||||
 | 
					            In_Data  => In_Data,
 | 
				
			||||||
 | 
					            In_Last  => In_Last,
 | 
				
			||||||
 | 
					            Out_Data => Out_Data,
 | 
				
			||||||
 | 
					            Out_Last => Out_Last,
 | 
				
			||||||
 | 
					            Flush    => Flush);
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					         Translate_Auto
 | 
				
			||||||
 | 
					           (Filter   => Filter,
 | 
				
			||||||
 | 
					            In_Data  => In_Data,
 | 
				
			||||||
 | 
					            In_Last  => In_Last,
 | 
				
			||||||
 | 
					            Out_Data => Out_Data,
 | 
				
			||||||
 | 
					            Out_Last => Out_Last,
 | 
				
			||||||
 | 
					            Flush    => Flush);
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Translate;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					   -- Translate_Auto --
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Translate_Auto
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      In_Data   : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      In_Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      use type Thin.Int;
 | 
				
			||||||
 | 
					      Code : Thin.Int;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if not Is_Open (Filter) then
 | 
				
			||||||
 | 
					         raise Status_Error;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Out_Data'Length = 0 and then In_Data'Length = 0 then
 | 
				
			||||||
 | 
					         raise Constraint_Error;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Set_Out (Filter.Strm.all, Out_Data'Address, Out_Data'Length);
 | 
				
			||||||
 | 
					      Set_In  (Filter.Strm.all, In_Data'Address, In_Data'Length);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Code := Flate (Filter.Compression).Step
 | 
				
			||||||
 | 
					        (To_Thin_Access (Filter.Strm),
 | 
				
			||||||
 | 
					         Thin.Int (Flush));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Code = Thin.Z_STREAM_END then
 | 
				
			||||||
 | 
					         Filter.Stream_End := True;
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					         Check_Error (Filter.Strm.all, Code);
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      In_Last  := In_Data'Last
 | 
				
			||||||
 | 
					         - Stream_Element_Offset (Avail_In (Filter.Strm.all));
 | 
				
			||||||
 | 
					      Out_Last := Out_Data'Last
 | 
				
			||||||
 | 
					         - Stream_Element_Offset (Avail_Out (Filter.Strm.all));
 | 
				
			||||||
 | 
					   end Translate_Auto;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					   -- Translate_GZip --
 | 
				
			||||||
 | 
					   --------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Translate_GZip
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      In_Data   : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      In_Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      Out_First : Stream_Element_Offset;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Add_Data (Data : in Stream_Element_Array);
 | 
				
			||||||
 | 
					      --  Add data to stream from the Filter.Offset till necessary,
 | 
				
			||||||
 | 
					      --  used for add gzip headr/footer.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Put_32
 | 
				
			||||||
 | 
					        (Item : in out Stream_Element_Array;
 | 
				
			||||||
 | 
					         Data : in     Unsigned_32);
 | 
				
			||||||
 | 
					      pragma Inline (Put_32);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      --------------
 | 
				
			||||||
 | 
					      -- Add_Data --
 | 
				
			||||||
 | 
					      --------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Add_Data (Data : in Stream_Element_Array) is
 | 
				
			||||||
 | 
					         Data_First : Stream_Element_Offset renames Filter.Offset;
 | 
				
			||||||
 | 
					         Data_Last  : Stream_Element_Offset;
 | 
				
			||||||
 | 
					         Data_Len   : Stream_Element_Offset; --  -1
 | 
				
			||||||
 | 
					         Out_Len    : Stream_Element_Offset; --  -1
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         Out_First := Out_Last + 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         if Data_First > Data'Last then
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Data_Len  := Data'Last     - Data_First;
 | 
				
			||||||
 | 
					         Out_Len   := Out_Data'Last - Out_First;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         if Data_Len <= Out_Len then
 | 
				
			||||||
 | 
					            Out_Last  := Out_First  + Data_Len;
 | 
				
			||||||
 | 
					            Data_Last := Data'Last;
 | 
				
			||||||
 | 
					         else
 | 
				
			||||||
 | 
					            Out_Last  := Out_Data'Last;
 | 
				
			||||||
 | 
					            Data_Last := Data_First + Out_Len;
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Out_Data (Out_First .. Out_Last) := Data (Data_First .. Data_Last);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Data_First := Data_Last + 1;
 | 
				
			||||||
 | 
					         Out_First  := Out_Last + 1;
 | 
				
			||||||
 | 
					      end Add_Data;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      ------------
 | 
				
			||||||
 | 
					      -- Put_32 --
 | 
				
			||||||
 | 
					      ------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      procedure Put_32
 | 
				
			||||||
 | 
					        (Item : in out Stream_Element_Array;
 | 
				
			||||||
 | 
					         Data : in     Unsigned_32)
 | 
				
			||||||
 | 
					      is
 | 
				
			||||||
 | 
					         D : Unsigned_32 := Data;
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					         for J in Item'First .. Item'First + 3 loop
 | 
				
			||||||
 | 
					            Item (J) := Stream_Element (D and 16#FF#);
 | 
				
			||||||
 | 
					            D := Shift_Right (D, 8);
 | 
				
			||||||
 | 
					         end loop;
 | 
				
			||||||
 | 
					      end Put_32;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      Out_Last := Out_Data'First - 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if not Filter.Stream_End then
 | 
				
			||||||
 | 
					         Add_Data (Simple_GZip_Header);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         Translate_Auto
 | 
				
			||||||
 | 
					           (Filter   => Filter,
 | 
				
			||||||
 | 
					            In_Data  => In_Data,
 | 
				
			||||||
 | 
					            In_Last  => In_Last,
 | 
				
			||||||
 | 
					            Out_Data => Out_Data (Out_First .. Out_Data'Last),
 | 
				
			||||||
 | 
					            Out_Last => Out_Last,
 | 
				
			||||||
 | 
					            Flush    => Flush);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         CRC32 (Filter.CRC, In_Data (In_Data'First .. In_Last));
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if Filter.Stream_End and then Out_Last <= Out_Data'Last then
 | 
				
			||||||
 | 
					         --  This detection method would work only when
 | 
				
			||||||
 | 
					         --  Simple_GZip_Header'Last > Footer_Array'Last
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         if Filter.Offset = Simple_GZip_Header'Last + 1 then
 | 
				
			||||||
 | 
					            Filter.Offset := Footer_Array'First;
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         declare
 | 
				
			||||||
 | 
					            Footer : Footer_Array;
 | 
				
			||||||
 | 
					         begin
 | 
				
			||||||
 | 
					            Put_32 (Footer, Filter.CRC);
 | 
				
			||||||
 | 
					            Put_32 (Footer (Footer'First + 4 .. Footer'Last),
 | 
				
			||||||
 | 
					                    Unsigned_32 (Total_In (Filter)));
 | 
				
			||||||
 | 
					            Add_Data (Footer);
 | 
				
			||||||
 | 
					         end;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					   end Translate_GZip;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					   -- Version --
 | 
				
			||||||
 | 
					   -------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Version return String is
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      return Interfaces.C.Strings.Value (Thin.zlibVersion);
 | 
				
			||||||
 | 
					   end Version;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					   -- Write --
 | 
				
			||||||
 | 
					   -----------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Write
 | 
				
			||||||
 | 
					     (Filter : in out Filter_Type;
 | 
				
			||||||
 | 
					      Item   : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Flush  : in     Flush_Mode := No_Flush)
 | 
				
			||||||
 | 
					   is
 | 
				
			||||||
 | 
					      Buffer   : Stream_Element_Array (1 .. Buffer_Size);
 | 
				
			||||||
 | 
					      In_Last  : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Last : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      In_First : Stream_Element_Offset := Item'First;
 | 
				
			||||||
 | 
					   begin
 | 
				
			||||||
 | 
					      if Item'Length = 0 and Flush = No_Flush then
 | 
				
			||||||
 | 
					         return;
 | 
				
			||||||
 | 
					      end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      loop
 | 
				
			||||||
 | 
					         Translate
 | 
				
			||||||
 | 
					           (Filter   => Filter,
 | 
				
			||||||
 | 
					            In_Data  => Item (In_First .. Item'Last),
 | 
				
			||||||
 | 
					            In_Last  => In_Last,
 | 
				
			||||||
 | 
					            Out_Data => Buffer,
 | 
				
			||||||
 | 
					            Out_Last => Out_Last,
 | 
				
			||||||
 | 
					            Flush    => Flush);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         if Out_Last >= Buffer'First then
 | 
				
			||||||
 | 
					            Write (Buffer (1 .. Out_Last));
 | 
				
			||||||
 | 
					         end if;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         exit when In_Last = Item'Last or Stream_End (Filter);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         In_First := In_Last + 1;
 | 
				
			||||||
 | 
					      end loop;
 | 
				
			||||||
 | 
					   end Write;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end ZLib;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,328 @@
 | 
				
			||||||
 | 
					------------------------------------------------------------------------------
 | 
				
			||||||
 | 
					--                      ZLib for Ada thick binding.                         --
 | 
				
			||||||
 | 
					--                                                                          --
 | 
				
			||||||
 | 
					--              Copyright (C) 2002-2004 Dmitriy Anisimkov                   --
 | 
				
			||||||
 | 
					--                                                                          --
 | 
				
			||||||
 | 
					--  This library is free software; you can redistribute it and/or modify    --
 | 
				
			||||||
 | 
					--  it under the terms of the GNU General Public License as published by    --
 | 
				
			||||||
 | 
					--  the Free Software Foundation; either version 2 of the License, or (at   --
 | 
				
			||||||
 | 
					--  your option) any later version.                                         --
 | 
				
			||||||
 | 
					--                                                                          --
 | 
				
			||||||
 | 
					--  This library is distributed in the hope that it will be useful, but     --
 | 
				
			||||||
 | 
					--  WITHOUT ANY WARRANTY; without even the implied warranty of              --
 | 
				
			||||||
 | 
					--  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU       --
 | 
				
			||||||
 | 
					--  General Public License for more details.                                --
 | 
				
			||||||
 | 
					--                                                                          --
 | 
				
			||||||
 | 
					--  You should have received a copy of the GNU General Public License       --
 | 
				
			||||||
 | 
					--  along with this library; if not, write to the Free Software Foundation, --
 | 
				
			||||||
 | 
					--  Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.          --
 | 
				
			||||||
 | 
					--                                                                          --
 | 
				
			||||||
 | 
					--  As a special exception, if other files instantiate generics from this   --
 | 
				
			||||||
 | 
					--  unit, or you link this unit with other files to produce an executable,  --
 | 
				
			||||||
 | 
					--  this  unit  does not  by itself cause  the resulting executable to be   --
 | 
				
			||||||
 | 
					--  covered by the GNU General Public License. This exception does not      --
 | 
				
			||||||
 | 
					--  however invalidate any other reasons why the executable file  might be  --
 | 
				
			||||||
 | 
					--  covered by the  GNU Public License.                                     --
 | 
				
			||||||
 | 
					------------------------------------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--  $Id: zlib.ads,v 1.26 2004/09/06 06:53:19 vagul Exp $
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Ada.Streams;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					with Interfaces;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					package ZLib is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ZLib_Error   : exception;
 | 
				
			||||||
 | 
					   Status_Error : exception;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Compression_Level is new Integer range -1 .. 9;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Flush_Mode is private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Compression_Method is private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Window_Bits_Type is new Integer range 8 .. 15;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Memory_Level_Type is new Integer range 1 .. 9;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Unsigned_32 is new Interfaces.Unsigned_32;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Strategy_Type is private;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Header_Type is (None, Auto, Default, GZip);
 | 
				
			||||||
 | 
					   --  Header type usage have a some limitation for inflate.
 | 
				
			||||||
 | 
					   --  See comment for Inflate_Init.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   subtype Count is Ada.Streams.Stream_Element_Count;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Default_Memory_Level : constant Memory_Level_Type := 8;
 | 
				
			||||||
 | 
					   Default_Window_Bits  : constant Window_Bits_Type  := 15;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ----------------------------------
 | 
				
			||||||
 | 
					   -- Compression method constants --
 | 
				
			||||||
 | 
					   ----------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Deflated : constant Compression_Method;
 | 
				
			||||||
 | 
					   --  Only one method allowed in this ZLib version
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ---------------------------------
 | 
				
			||||||
 | 
					   -- Compression level constants --
 | 
				
			||||||
 | 
					   ---------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   No_Compression      : constant Compression_Level := 0;
 | 
				
			||||||
 | 
					   Best_Speed          : constant Compression_Level := 1;
 | 
				
			||||||
 | 
					   Best_Compression    : constant Compression_Level := 9;
 | 
				
			||||||
 | 
					   Default_Compression : constant Compression_Level := -1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --------------------------
 | 
				
			||||||
 | 
					   -- Flush mode constants --
 | 
				
			||||||
 | 
					   --------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   No_Flush      : constant Flush_Mode;
 | 
				
			||||||
 | 
					   --  Regular way for compression, no flush
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Partial_Flush : constant Flush_Mode;
 | 
				
			||||||
 | 
					   --  Will be removed, use Z_SYNC_FLUSH instead
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Sync_Flush    : constant Flush_Mode;
 | 
				
			||||||
 | 
					   --  All pending output is flushed to the output buffer and the output
 | 
				
			||||||
 | 
					   --  is aligned on a byte boundary, so that the decompressor can get all
 | 
				
			||||||
 | 
					   --  input data available so far. (In particular avail_in is zero after the
 | 
				
			||||||
 | 
					   --  call if enough output space has been provided  before the call.)
 | 
				
			||||||
 | 
					   --  Flushing may degrade compression for some compression algorithms and so
 | 
				
			||||||
 | 
					   --  it should be used only when necessary.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Block_Flush   : constant Flush_Mode;
 | 
				
			||||||
 | 
					   --  Z_BLOCK requests that inflate() stop
 | 
				
			||||||
 | 
					   --  if and when it get to the next deflate block boundary. When decoding the
 | 
				
			||||||
 | 
					   --  zlib or gzip format, this will cause inflate() to return immediately
 | 
				
			||||||
 | 
					   --  after the header and before the first block. When doing a raw inflate,
 | 
				
			||||||
 | 
					   --  inflate() will go ahead and process the first block, and will return
 | 
				
			||||||
 | 
					   --  when it gets to the end of that block, or when it runs out of data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Full_Flush    : constant Flush_Mode;
 | 
				
			||||||
 | 
					   --  All output is flushed as with SYNC_FLUSH, and the compression state
 | 
				
			||||||
 | 
					   --  is reset so that decompression can restart from this point if previous
 | 
				
			||||||
 | 
					   --  compressed data has been damaged or if random access is desired. Using
 | 
				
			||||||
 | 
					   --  Full_Flush too often can seriously degrade the compression.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Finish        : constant Flush_Mode;
 | 
				
			||||||
 | 
					   --  Just for tell the compressor that input data is complete.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   ------------------------------------
 | 
				
			||||||
 | 
					   -- Compression strategy constants --
 | 
				
			||||||
 | 
					   ------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   --  RLE strategy could be used only in version 1.2.0 and later.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Filtered         : constant Strategy_Type;
 | 
				
			||||||
 | 
					   Huffman_Only     : constant Strategy_Type;
 | 
				
			||||||
 | 
					   RLE              : constant Strategy_Type;
 | 
				
			||||||
 | 
					   Default_Strategy : constant Strategy_Type;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Default_Buffer_Size : constant := 4096;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Filter_Type is tagged limited private;
 | 
				
			||||||
 | 
					   --  The filter is for compression and for decompression.
 | 
				
			||||||
 | 
					   --  The usage of the type is depend of its initialization.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Version return String;
 | 
				
			||||||
 | 
					   pragma Inline (Version);
 | 
				
			||||||
 | 
					   --  Return string representation of the ZLib version.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Deflate_Init
 | 
				
			||||||
 | 
					     (Filter       : in out Filter_Type;
 | 
				
			||||||
 | 
					      Level        : in     Compression_Level  := Default_Compression;
 | 
				
			||||||
 | 
					      Strategy     : in     Strategy_Type      := Default_Strategy;
 | 
				
			||||||
 | 
					      Method       : in     Compression_Method := Deflated;
 | 
				
			||||||
 | 
					      Window_Bits  : in     Window_Bits_Type   := Default_Window_Bits;
 | 
				
			||||||
 | 
					      Memory_Level : in     Memory_Level_Type  := Default_Memory_Level;
 | 
				
			||||||
 | 
					      Header       : in     Header_Type        := Default);
 | 
				
			||||||
 | 
					   --  Compressor initialization.
 | 
				
			||||||
 | 
					   --  When Header parameter is Auto or Default, then default zlib header
 | 
				
			||||||
 | 
					   --  would be provided for compressed data.
 | 
				
			||||||
 | 
					   --  When Header is GZip, then gzip header would be set instead of
 | 
				
			||||||
 | 
					   --  default header.
 | 
				
			||||||
 | 
					   --  When Header is None, no header would be set for compressed data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Inflate_Init
 | 
				
			||||||
 | 
					     (Filter      : in out Filter_Type;
 | 
				
			||||||
 | 
					      Window_Bits : in     Window_Bits_Type := Default_Window_Bits;
 | 
				
			||||||
 | 
					      Header      : in     Header_Type      := Default);
 | 
				
			||||||
 | 
					   --  Decompressor initialization.
 | 
				
			||||||
 | 
					   --  Default header type mean that ZLib default header is expecting in the
 | 
				
			||||||
 | 
					   --  input compressed stream.
 | 
				
			||||||
 | 
					   --  Header type None mean that no header is expecting in the input stream.
 | 
				
			||||||
 | 
					   --  GZip header type mean that GZip header is expecting in the
 | 
				
			||||||
 | 
					   --  input compressed stream.
 | 
				
			||||||
 | 
					   --  Auto header type mean that header type (GZip or Native) would be
 | 
				
			||||||
 | 
					   --  detected automatically in the input stream.
 | 
				
			||||||
 | 
					   --  Note that header types parameter values None, GZip and Auto are
 | 
				
			||||||
 | 
					   --  supported for inflate routine only in ZLib versions 1.2.0.2 and later.
 | 
				
			||||||
 | 
					   --  Deflate_Init is supporting all header types.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Is_Open (Filter : in Filter_Type) return Boolean;
 | 
				
			||||||
 | 
					   pragma Inline (Is_Open);
 | 
				
			||||||
 | 
					   --  Is the filter opened for compression or decompression.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Close
 | 
				
			||||||
 | 
					     (Filter       : in out Filter_Type;
 | 
				
			||||||
 | 
					      Ignore_Error : in     Boolean := False);
 | 
				
			||||||
 | 
					   --  Closing the compression or decompressor.
 | 
				
			||||||
 | 
					   --  If stream is closing before the complete and Ignore_Error is False,
 | 
				
			||||||
 | 
					   --  The exception would be raised.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   generic
 | 
				
			||||||
 | 
					      with procedure Data_In
 | 
				
			||||||
 | 
					        (Item : out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					         Last : out Ada.Streams.Stream_Element_Offset);
 | 
				
			||||||
 | 
					      with procedure Data_Out
 | 
				
			||||||
 | 
					        (Item : in Ada.Streams.Stream_Element_Array);
 | 
				
			||||||
 | 
					   procedure Generic_Translate
 | 
				
			||||||
 | 
					     (Filter          : in out Filter_Type;
 | 
				
			||||||
 | 
					      In_Buffer_Size  : in     Integer := Default_Buffer_Size;
 | 
				
			||||||
 | 
					      Out_Buffer_Size : in     Integer := Default_Buffer_Size);
 | 
				
			||||||
 | 
					   --  Compress/decompress data fetch from Data_In routine and pass the result
 | 
				
			||||||
 | 
					   --  to the Data_Out routine. User should provide Data_In and Data_Out
 | 
				
			||||||
 | 
					   --  for compression/decompression data flow.
 | 
				
			||||||
 | 
					   --  Compression or decompression depend on Filter initialization.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_In (Filter : in Filter_Type) return Count;
 | 
				
			||||||
 | 
					   pragma Inline (Total_In);
 | 
				
			||||||
 | 
					   --  Returns total number of input bytes read so far
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Total_Out (Filter : in Filter_Type) return Count;
 | 
				
			||||||
 | 
					   pragma Inline (Total_Out);
 | 
				
			||||||
 | 
					   --  Returns total number of bytes output so far
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function CRC32
 | 
				
			||||||
 | 
					     (CRC    : in Unsigned_32;
 | 
				
			||||||
 | 
					      Data   : in Ada.Streams.Stream_Element_Array)
 | 
				
			||||||
 | 
					      return Unsigned_32;
 | 
				
			||||||
 | 
					   pragma Inline (CRC32);
 | 
				
			||||||
 | 
					   --  Compute CRC32, it could be necessary for make gzip format
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure CRC32
 | 
				
			||||||
 | 
					     (CRC  : in out Unsigned_32;
 | 
				
			||||||
 | 
					      Data : in     Ada.Streams.Stream_Element_Array);
 | 
				
			||||||
 | 
					   pragma Inline (CRC32);
 | 
				
			||||||
 | 
					   --  Compute CRC32, it could be necessary for make gzip format
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   -------------------------------------------------
 | 
				
			||||||
 | 
					   --  Below is more complex low level routines.  --
 | 
				
			||||||
 | 
					   -------------------------------------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Translate
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      In_Data   : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      In_Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode);
 | 
				
			||||||
 | 
					   --  Compress/decompress the In_Data buffer and place the result into
 | 
				
			||||||
 | 
					   --  Out_Data. In_Last is the index of last element from In_Data accepted by
 | 
				
			||||||
 | 
					   --  the Filter. Out_Last is the last element of the received data from
 | 
				
			||||||
 | 
					   --  Filter. To tell the filter that incoming data are complete put the
 | 
				
			||||||
 | 
					   --  Flush parameter to Finish.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   function Stream_End (Filter : in Filter_Type) return Boolean;
 | 
				
			||||||
 | 
					   pragma Inline (Stream_End);
 | 
				
			||||||
 | 
					   --  Return the true when the stream is complete.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Flush
 | 
				
			||||||
 | 
					     (Filter    : in out Filter_Type;
 | 
				
			||||||
 | 
					      Out_Data  :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Out_Last  :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush     : in     Flush_Mode);
 | 
				
			||||||
 | 
					   pragma Inline (Flush);
 | 
				
			||||||
 | 
					   --  Flushing the data from the compressor.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   generic
 | 
				
			||||||
 | 
					      with procedure Write
 | 
				
			||||||
 | 
					        (Item : in Ada.Streams.Stream_Element_Array);
 | 
				
			||||||
 | 
					      --  User should provide this routine for accept
 | 
				
			||||||
 | 
					      --  compressed/decompressed data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Buffer_Size : in Ada.Streams.Stream_Element_Offset
 | 
				
			||||||
 | 
					         := Default_Buffer_Size;
 | 
				
			||||||
 | 
					      --  Buffer size for Write user routine.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Write
 | 
				
			||||||
 | 
					     (Filter  : in out Filter_Type;
 | 
				
			||||||
 | 
					      Item    : in     Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Flush   : in     Flush_Mode := No_Flush);
 | 
				
			||||||
 | 
					   --  Compress/Decompress data from Item to the generic parameter procedure
 | 
				
			||||||
 | 
					   --  Write. Output buffer size could be set in Buffer_Size generic parameter.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   generic
 | 
				
			||||||
 | 
					      with procedure Read
 | 
				
			||||||
 | 
					        (Item : out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					         Last : out Ada.Streams.Stream_Element_Offset);
 | 
				
			||||||
 | 
					      --  User should provide data for compression/decompression
 | 
				
			||||||
 | 
					      --  thru this routine.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Buffer : in out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      --  Buffer for keep remaining data from the previous
 | 
				
			||||||
 | 
					      --  back read.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Rest_First, Rest_Last : in out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      --  Rest_First have to be initialized to Buffer'Last + 1
 | 
				
			||||||
 | 
					      --  Rest_Last have to be initialized to Buffer'Last
 | 
				
			||||||
 | 
					      --  before usage.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Allow_Read_Some : in Boolean := False;
 | 
				
			||||||
 | 
					      --  Is it allowed to return Last < Item'Last before end of data.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   procedure Read
 | 
				
			||||||
 | 
					     (Filter : in out Filter_Type;
 | 
				
			||||||
 | 
					      Item   :    out Ada.Streams.Stream_Element_Array;
 | 
				
			||||||
 | 
					      Last   :    out Ada.Streams.Stream_Element_Offset;
 | 
				
			||||||
 | 
					      Flush  : in     Flush_Mode := No_Flush);
 | 
				
			||||||
 | 
					   --  Compress/Decompress data from generic parameter procedure Read to the
 | 
				
			||||||
 | 
					   --  Item. User should provide Buffer and initialized Rest_First, Rest_Last
 | 
				
			||||||
 | 
					   --  indicators. If Allow_Read_Some is True, Read routines could return
 | 
				
			||||||
 | 
					   --  Last < Item'Last only at end of stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					private
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   use Ada.Streams;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   pragma Assert (Ada.Streams.Stream_Element'Size    =    8);
 | 
				
			||||||
 | 
					   pragma Assert (Ada.Streams.Stream_Element'Modulus = 2**8);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Flush_Mode is new Integer range 0 .. 5;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Compression_Method is new Integer range 8 .. 8;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Strategy_Type is new Integer range 0 .. 3;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   No_Flush      : constant Flush_Mode := 0;
 | 
				
			||||||
 | 
					   Partial_Flush : constant Flush_Mode := 1;
 | 
				
			||||||
 | 
					   Sync_Flush    : constant Flush_Mode := 2;
 | 
				
			||||||
 | 
					   Full_Flush    : constant Flush_Mode := 3;
 | 
				
			||||||
 | 
					   Finish        : constant Flush_Mode := 4;
 | 
				
			||||||
 | 
					   Block_Flush   : constant Flush_Mode := 5;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Filtered         : constant Strategy_Type := 1;
 | 
				
			||||||
 | 
					   Huffman_Only     : constant Strategy_Type := 2;
 | 
				
			||||||
 | 
					   RLE              : constant Strategy_Type := 3;
 | 
				
			||||||
 | 
					   Default_Strategy : constant Strategy_Type := 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Deflated : constant Compression_Method := 8;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Z_Stream;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Z_Stream_Access is access all Z_Stream;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   type Filter_Type is tagged limited record
 | 
				
			||||||
 | 
					      Strm        : Z_Stream_Access;
 | 
				
			||||||
 | 
					      Compression : Boolean;
 | 
				
			||||||
 | 
					      Stream_End  : Boolean;
 | 
				
			||||||
 | 
					      Header      : Header_Type;
 | 
				
			||||||
 | 
					      CRC         : Unsigned_32;
 | 
				
			||||||
 | 
					      Offset      : Stream_Element_Offset;
 | 
				
			||||||
 | 
					      --  Offset for gzip header/footer output.
 | 
				
			||||||
 | 
					   end record;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end ZLib;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,20 @@
 | 
				
			||||||
 | 
					project Zlib is
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   for Languages use ("Ada");
 | 
				
			||||||
 | 
					   for Source_Dirs use (".");
 | 
				
			||||||
 | 
					   for Object_Dir use ".";
 | 
				
			||||||
 | 
					   for Main use ("test.adb", "mtest.adb", "read.adb", "buffer_demo");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   package Compiler is
 | 
				
			||||||
 | 
					      for Default_Switches ("ada") use ("-gnatwcfilopru", "-gnatVcdfimorst", "-gnatyabcefhiklmnoprst");
 | 
				
			||||||
 | 
					   end Compiler;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   package Linker is
 | 
				
			||||||
 | 
					      for Default_Switches ("ada") use ("-lz");
 | 
				
			||||||
 | 
					   end Linker;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   package Builder is
 | 
				
			||||||
 | 
					      for Default_Switches ("ada") use ("-s", "-gnatQ");
 | 
				
			||||||
 | 
					   end Builder;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end Zlib;
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,8 @@
 | 
				
			||||||
 | 
					blast: blast.c blast.h
 | 
				
			||||||
 | 
						cc -DTEST -o blast blast.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					test: blast
 | 
				
			||||||
 | 
						blast < test.pk | cmp - test.txt
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					clean:
 | 
				
			||||||
 | 
						rm -f blast blast.o
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,4 @@
 | 
				
			||||||
 | 
					Read blast.h for purpose and usage.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Mark Adler
 | 
				
			||||||
 | 
					madler@alumni.caltech.edu
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,466 @@
 | 
				
			||||||
 | 
					/* blast.c
 | 
				
			||||||
 | 
					 * Copyright (C) 2003, 2012, 2013 Mark Adler
 | 
				
			||||||
 | 
					 * For conditions of distribution and use, see copyright notice in blast.h
 | 
				
			||||||
 | 
					 * version 1.3, 24 Aug 2013
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * blast.c decompresses data compressed by the PKWare Compression Library.
 | 
				
			||||||
 | 
					 * This function provides functionality similar to the explode() function of
 | 
				
			||||||
 | 
					 * the PKWare library, hence the name "blast".
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * This decompressor is based on the excellent format description provided by
 | 
				
			||||||
 | 
					 * Ben Rudiak-Gould in comp.compression on August 13, 2001.  Interestingly, the
 | 
				
			||||||
 | 
					 * example Ben provided in the post is incorrect.  The distance 110001 should
 | 
				
			||||||
 | 
					 * instead be 111000.  When corrected, the example byte stream becomes:
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 *    00 04 82 24 25 8f 80 7f
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * which decompresses to "AIAIAIAIAIAIA" (without the quotes).
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Change history:
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * 1.0  12 Feb 2003     - First version
 | 
				
			||||||
 | 
					 * 1.1  16 Feb 2003     - Fixed distance check for > 4 GB uncompressed data
 | 
				
			||||||
 | 
					 * 1.2  24 Oct 2012     - Add note about using binary mode in stdio
 | 
				
			||||||
 | 
					 *                      - Fix comparisons of differently signed integers
 | 
				
			||||||
 | 
					 * 1.3  24 Aug 2013     - Return unused input from blast()
 | 
				
			||||||
 | 
					 *                      - Fix test code to correctly report unused input
 | 
				
			||||||
 | 
					 *                      - Enable the provision of initial input to blast()
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <stddef.h>             /* for NULL */
 | 
				
			||||||
 | 
					#include <setjmp.h>             /* for setjmp(), longjmp(), and jmp_buf */
 | 
				
			||||||
 | 
					#include "blast.h"              /* prototype for blast() */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define local static            /* for local function definitions */
 | 
				
			||||||
 | 
					#define MAXBITS 13              /* maximum code length */
 | 
				
			||||||
 | 
					#define MAXWIN 4096             /* maximum window size */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* input and output state */
 | 
				
			||||||
 | 
					struct state {
 | 
				
			||||||
 | 
					    /* input state */
 | 
				
			||||||
 | 
					    blast_in infun;             /* input function provided by user */
 | 
				
			||||||
 | 
					    void *inhow;                /* opaque information passed to infun() */
 | 
				
			||||||
 | 
					    unsigned char *in;          /* next input location */
 | 
				
			||||||
 | 
					    unsigned left;              /* available input at in */
 | 
				
			||||||
 | 
					    int bitbuf;                 /* bit buffer */
 | 
				
			||||||
 | 
					    int bitcnt;                 /* number of bits in bit buffer */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* input limit error return state for bits() and decode() */
 | 
				
			||||||
 | 
					    jmp_buf env;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* output state */
 | 
				
			||||||
 | 
					    blast_out outfun;           /* output function provided by user */
 | 
				
			||||||
 | 
					    void *outhow;               /* opaque information passed to outfun() */
 | 
				
			||||||
 | 
					    unsigned next;              /* index of next write location in out[] */
 | 
				
			||||||
 | 
					    int first;                  /* true to check distances (for first 4K) */
 | 
				
			||||||
 | 
					    unsigned char out[MAXWIN];  /* output buffer and sliding window */
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Return need bits from the input stream.  This always leaves less than
 | 
				
			||||||
 | 
					 * eight bits in the buffer.  bits() works properly for need == 0.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Format notes:
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - Bits are stored in bytes from the least significant bit to the most
 | 
				
			||||||
 | 
					 *   significant bit.  Therefore bits are dropped from the bottom of the bit
 | 
				
			||||||
 | 
					 *   buffer, using shift right, and new bytes are appended to the top of the
 | 
				
			||||||
 | 
					 *   bit buffer, using shift left.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					local int bits(struct state *s, int need)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int val;            /* bit accumulator */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* load at least need bits into val */
 | 
				
			||||||
 | 
					    val = s->bitbuf;
 | 
				
			||||||
 | 
					    while (s->bitcnt < need) {
 | 
				
			||||||
 | 
					        if (s->left == 0) {
 | 
				
			||||||
 | 
					            s->left = s->infun(s->inhow, &(s->in));
 | 
				
			||||||
 | 
					            if (s->left == 0) longjmp(s->env, 1);       /* out of input */
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        val |= (int)(*(s->in)++) << s->bitcnt;          /* load eight bits */
 | 
				
			||||||
 | 
					        s->left--;
 | 
				
			||||||
 | 
					        s->bitcnt += 8;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* drop need bits and update buffer, always zero to seven bits left */
 | 
				
			||||||
 | 
					    s->bitbuf = val >> need;
 | 
				
			||||||
 | 
					    s->bitcnt -= need;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* return need bits, zeroing the bits above that */
 | 
				
			||||||
 | 
					    return val & ((1 << need) - 1);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Huffman code decoding tables.  count[1..MAXBITS] is the number of symbols of
 | 
				
			||||||
 | 
					 * each length, which for a canonical code are stepped through in order.
 | 
				
			||||||
 | 
					 * symbol[] are the symbol values in canonical order, where the number of
 | 
				
			||||||
 | 
					 * entries is the sum of the counts in count[].  The decoding process can be
 | 
				
			||||||
 | 
					 * seen in the function decode() below.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					struct huffman {
 | 
				
			||||||
 | 
					    short *count;       /* number of symbols of each length */
 | 
				
			||||||
 | 
					    short *symbol;      /* canonically ordered symbols */
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Decode a code from the stream s using huffman table h.  Return the symbol or
 | 
				
			||||||
 | 
					 * a negative value if there is an error.  If all of the lengths are zero, i.e.
 | 
				
			||||||
 | 
					 * an empty code, or if the code is incomplete and an invalid code is received,
 | 
				
			||||||
 | 
					 * then -9 is returned after reading MAXBITS bits.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Format notes:
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - The codes as stored in the compressed data are bit-reversed relative to
 | 
				
			||||||
 | 
					 *   a simple integer ordering of codes of the same lengths.  Hence below the
 | 
				
			||||||
 | 
					 *   bits are pulled from the compressed data one at a time and used to
 | 
				
			||||||
 | 
					 *   build the code value reversed from what is in the stream in order to
 | 
				
			||||||
 | 
					 *   permit simple integer comparisons for decoding.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - The first code for the shortest length is all ones.  Subsequent codes of
 | 
				
			||||||
 | 
					 *   the same length are simply integer decrements of the previous code.  When
 | 
				
			||||||
 | 
					 *   moving up a length, a one bit is appended to the code.  For a complete
 | 
				
			||||||
 | 
					 *   code, the last code of the longest length will be all zeros.  To support
 | 
				
			||||||
 | 
					 *   this ordering, the bits pulled during decoding are inverted to apply the
 | 
				
			||||||
 | 
					 *   more "natural" ordering starting with all zeros and incrementing.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					local int decode(struct state *s, struct huffman *h)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int len;            /* current number of bits in code */
 | 
				
			||||||
 | 
					    int code;           /* len bits being decoded */
 | 
				
			||||||
 | 
					    int first;          /* first code of length len */
 | 
				
			||||||
 | 
					    int count;          /* number of codes of length len */
 | 
				
			||||||
 | 
					    int index;          /* index of first code of length len in symbol table */
 | 
				
			||||||
 | 
					    int bitbuf;         /* bits from stream */
 | 
				
			||||||
 | 
					    int left;           /* bits left in next or left to process */
 | 
				
			||||||
 | 
					    short *next;        /* next number of codes */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    bitbuf = s->bitbuf;
 | 
				
			||||||
 | 
					    left = s->bitcnt;
 | 
				
			||||||
 | 
					    code = first = index = 0;
 | 
				
			||||||
 | 
					    len = 1;
 | 
				
			||||||
 | 
					    next = h->count + 1;
 | 
				
			||||||
 | 
					    while (1) {
 | 
				
			||||||
 | 
					        while (left--) {
 | 
				
			||||||
 | 
					            code |= (bitbuf & 1) ^ 1;   /* invert code */
 | 
				
			||||||
 | 
					            bitbuf >>= 1;
 | 
				
			||||||
 | 
					            count = *next++;
 | 
				
			||||||
 | 
					            if (code < first + count) { /* if length len, return symbol */
 | 
				
			||||||
 | 
					                s->bitbuf = bitbuf;
 | 
				
			||||||
 | 
					                s->bitcnt = (s->bitcnt - len) & 7;
 | 
				
			||||||
 | 
					                return h->symbol[index + (code - first)];
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            index += count;             /* else update for next length */
 | 
				
			||||||
 | 
					            first += count;
 | 
				
			||||||
 | 
					            first <<= 1;
 | 
				
			||||||
 | 
					            code <<= 1;
 | 
				
			||||||
 | 
					            len++;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        left = (MAXBITS+1) - len;
 | 
				
			||||||
 | 
					        if (left == 0) break;
 | 
				
			||||||
 | 
					        if (s->left == 0) {
 | 
				
			||||||
 | 
					            s->left = s->infun(s->inhow, &(s->in));
 | 
				
			||||||
 | 
					            if (s->left == 0) longjmp(s->env, 1);       /* out of input */
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        bitbuf = *(s->in)++;
 | 
				
			||||||
 | 
					        s->left--;
 | 
				
			||||||
 | 
					        if (left > 8) left = 8;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return -9;                          /* ran out of codes */
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Given a list of repeated code lengths rep[0..n-1], where each byte is a
 | 
				
			||||||
 | 
					 * count (high four bits + 1) and a code length (low four bits), generate the
 | 
				
			||||||
 | 
					 * list of code lengths.  This compaction reduces the size of the object code.
 | 
				
			||||||
 | 
					 * Then given the list of code lengths length[0..n-1] representing a canonical
 | 
				
			||||||
 | 
					 * Huffman code for n symbols, construct the tables required to decode those
 | 
				
			||||||
 | 
					 * codes.  Those tables are the number of codes of each length, and the symbols
 | 
				
			||||||
 | 
					 * sorted by length, retaining their original order within each length.  The
 | 
				
			||||||
 | 
					 * return value is zero for a complete code set, negative for an over-
 | 
				
			||||||
 | 
					 * subscribed code set, and positive for an incomplete code set.  The tables
 | 
				
			||||||
 | 
					 * can be used if the return value is zero or positive, but they cannot be used
 | 
				
			||||||
 | 
					 * if the return value is negative.  If the return value is zero, it is not
 | 
				
			||||||
 | 
					 * possible for decode() using that table to return an error--any stream of
 | 
				
			||||||
 | 
					 * enough bits will resolve to a symbol.  If the return value is positive, then
 | 
				
			||||||
 | 
					 * it is possible for decode() using that table to return an error for received
 | 
				
			||||||
 | 
					 * codes past the end of the incomplete lengths.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					local int construct(struct huffman *h, const unsigned char *rep, int n)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int symbol;         /* current symbol when stepping through length[] */
 | 
				
			||||||
 | 
					    int len;            /* current length when stepping through h->count[] */
 | 
				
			||||||
 | 
					    int left;           /* number of possible codes left of current length */
 | 
				
			||||||
 | 
					    short offs[MAXBITS+1];      /* offsets in symbol table for each length */
 | 
				
			||||||
 | 
					    short length[256];  /* code lengths */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* convert compact repeat counts into symbol bit length list */
 | 
				
			||||||
 | 
					    symbol = 0;
 | 
				
			||||||
 | 
					    do {
 | 
				
			||||||
 | 
					        len = *rep++;
 | 
				
			||||||
 | 
					        left = (len >> 4) + 1;
 | 
				
			||||||
 | 
					        len &= 15;
 | 
				
			||||||
 | 
					        do {
 | 
				
			||||||
 | 
					            length[symbol++] = len;
 | 
				
			||||||
 | 
					        } while (--left);
 | 
				
			||||||
 | 
					    } while (--n);
 | 
				
			||||||
 | 
					    n = symbol;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* count number of codes of each length */
 | 
				
			||||||
 | 
					    for (len = 0; len <= MAXBITS; len++)
 | 
				
			||||||
 | 
					        h->count[len] = 0;
 | 
				
			||||||
 | 
					    for (symbol = 0; symbol < n; symbol++)
 | 
				
			||||||
 | 
					        (h->count[length[symbol]])++;   /* assumes lengths are within bounds */
 | 
				
			||||||
 | 
					    if (h->count[0] == n)               /* no codes! */
 | 
				
			||||||
 | 
					        return 0;                       /* complete, but decode() will fail */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* check for an over-subscribed or incomplete set of lengths */
 | 
				
			||||||
 | 
					    left = 1;                           /* one possible code of zero length */
 | 
				
			||||||
 | 
					    for (len = 1; len <= MAXBITS; len++) {
 | 
				
			||||||
 | 
					        left <<= 1;                     /* one more bit, double codes left */
 | 
				
			||||||
 | 
					        left -= h->count[len];          /* deduct count from possible codes */
 | 
				
			||||||
 | 
					        if (left < 0) return left;      /* over-subscribed--return negative */
 | 
				
			||||||
 | 
					    }                                   /* left > 0 means incomplete */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* generate offsets into symbol table for each length for sorting */
 | 
				
			||||||
 | 
					    offs[1] = 0;
 | 
				
			||||||
 | 
					    for (len = 1; len < MAXBITS; len++)
 | 
				
			||||||
 | 
					        offs[len + 1] = offs[len] + h->count[len];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /*
 | 
				
			||||||
 | 
					     * put symbols in table sorted by length, by symbol order within each
 | 
				
			||||||
 | 
					     * length
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    for (symbol = 0; symbol < n; symbol++)
 | 
				
			||||||
 | 
					        if (length[symbol] != 0)
 | 
				
			||||||
 | 
					            h->symbol[offs[length[symbol]]++] = symbol;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* return zero for complete set, positive for incomplete set */
 | 
				
			||||||
 | 
					    return left;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Decode PKWare Compression Library stream.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Format notes:
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - First byte is 0 if literals are uncoded or 1 if they are coded.  Second
 | 
				
			||||||
 | 
					 *   byte is 4, 5, or 6 for the number of extra bits in the distance code.
 | 
				
			||||||
 | 
					 *   This is the base-2 logarithm of the dictionary size minus six.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - Compressed data is a combination of literals and length/distance pairs
 | 
				
			||||||
 | 
					 *   terminated by an end code.  Literals are either Huffman coded or
 | 
				
			||||||
 | 
					 *   uncoded bytes.  A length/distance pair is a coded length followed by a
 | 
				
			||||||
 | 
					 *   coded distance to represent a string that occurs earlier in the
 | 
				
			||||||
 | 
					 *   uncompressed data that occurs again at the current location.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - A bit preceding a literal or length/distance pair indicates which comes
 | 
				
			||||||
 | 
					 *   next, 0 for literals, 1 for length/distance.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - If literals are uncoded, then the next eight bits are the literal, in the
 | 
				
			||||||
 | 
					 *   normal bit order in the stream, i.e. no bit-reversal is needed. Similarly,
 | 
				
			||||||
 | 
					 *   no bit reversal is needed for either the length extra bits or the distance
 | 
				
			||||||
 | 
					 *   extra bits.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - Literal bytes are simply written to the output.  A length/distance pair is
 | 
				
			||||||
 | 
					 *   an instruction to copy previously uncompressed bytes to the output.  The
 | 
				
			||||||
 | 
					 *   copy is from distance bytes back in the output stream, copying for length
 | 
				
			||||||
 | 
					 *   bytes.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - Distances pointing before the beginning of the output data are not
 | 
				
			||||||
 | 
					 *   permitted.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * - Overlapped copies, where the length is greater than the distance, are
 | 
				
			||||||
 | 
					 *   allowed and common.  For example, a distance of one and a length of 518
 | 
				
			||||||
 | 
					 *   simply copies the last byte 518 times.  A distance of four and a length of
 | 
				
			||||||
 | 
					 *   twelve copies the last four bytes three times.  A simple forward copy
 | 
				
			||||||
 | 
					 *   ignoring whether the length is greater than the distance or not implements
 | 
				
			||||||
 | 
					 *   this correctly.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					local int decomp(struct state *s)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int lit;            /* true if literals are coded */
 | 
				
			||||||
 | 
					    int dict;           /* log2(dictionary size) - 6 */
 | 
				
			||||||
 | 
					    int symbol;         /* decoded symbol, extra bits for distance */
 | 
				
			||||||
 | 
					    int len;            /* length for copy */
 | 
				
			||||||
 | 
					    unsigned dist;      /* distance for copy */
 | 
				
			||||||
 | 
					    int copy;           /* copy counter */
 | 
				
			||||||
 | 
					    unsigned char *from, *to;   /* copy pointers */
 | 
				
			||||||
 | 
					    static int virgin = 1;                              /* build tables once */
 | 
				
			||||||
 | 
					    static short litcnt[MAXBITS+1], litsym[256];        /* litcode memory */
 | 
				
			||||||
 | 
					    static short lencnt[MAXBITS+1], lensym[16];         /* lencode memory */
 | 
				
			||||||
 | 
					    static short distcnt[MAXBITS+1], distsym[64];       /* distcode memory */
 | 
				
			||||||
 | 
					    static struct huffman litcode = {litcnt, litsym};   /* length code */
 | 
				
			||||||
 | 
					    static struct huffman lencode = {lencnt, lensym};   /* length code */
 | 
				
			||||||
 | 
					    static struct huffman distcode = {distcnt, distsym};/* distance code */
 | 
				
			||||||
 | 
					        /* bit lengths of literal codes */
 | 
				
			||||||
 | 
					    static const unsigned char litlen[] = {
 | 
				
			||||||
 | 
					        11, 124, 8, 7, 28, 7, 188, 13, 76, 4, 10, 8, 12, 10, 12, 10, 8, 23, 8,
 | 
				
			||||||
 | 
					        9, 7, 6, 7, 8, 7, 6, 55, 8, 23, 24, 12, 11, 7, 9, 11, 12, 6, 7, 22, 5,
 | 
				
			||||||
 | 
					        7, 24, 6, 11, 9, 6, 7, 22, 7, 11, 38, 7, 9, 8, 25, 11, 8, 11, 9, 12,
 | 
				
			||||||
 | 
					        8, 12, 5, 38, 5, 38, 5, 11, 7, 5, 6, 21, 6, 10, 53, 8, 7, 24, 10, 27,
 | 
				
			||||||
 | 
					        44, 253, 253, 253, 252, 252, 252, 13, 12, 45, 12, 45, 12, 61, 12, 45,
 | 
				
			||||||
 | 
					        44, 173};
 | 
				
			||||||
 | 
					        /* bit lengths of length codes 0..15 */
 | 
				
			||||||
 | 
					    static const unsigned char lenlen[] = {2, 35, 36, 53, 38, 23};
 | 
				
			||||||
 | 
					        /* bit lengths of distance codes 0..63 */
 | 
				
			||||||
 | 
					    static const unsigned char distlen[] = {2, 20, 53, 230, 247, 151, 248};
 | 
				
			||||||
 | 
					    static const short base[16] = {     /* base for length codes */
 | 
				
			||||||
 | 
					        3, 2, 4, 5, 6, 7, 8, 9, 10, 12, 16, 24, 40, 72, 136, 264};
 | 
				
			||||||
 | 
					    static const char extra[16] = {     /* extra bits for length codes */
 | 
				
			||||||
 | 
					        0, 0, 0, 0, 0, 0, 0, 0, 1, 2, 3, 4, 5, 6, 7, 8};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* set up decoding tables (once--might not be thread-safe) */
 | 
				
			||||||
 | 
					    if (virgin) {
 | 
				
			||||||
 | 
					        construct(&litcode, litlen, sizeof(litlen));
 | 
				
			||||||
 | 
					        construct(&lencode, lenlen, sizeof(lenlen));
 | 
				
			||||||
 | 
					        construct(&distcode, distlen, sizeof(distlen));
 | 
				
			||||||
 | 
					        virgin = 0;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* read header */
 | 
				
			||||||
 | 
					    lit = bits(s, 8);
 | 
				
			||||||
 | 
					    if (lit > 1) return -1;
 | 
				
			||||||
 | 
					    dict = bits(s, 8);
 | 
				
			||||||
 | 
					    if (dict < 4 || dict > 6) return -2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* decode literals and length/distance pairs */
 | 
				
			||||||
 | 
					    do {
 | 
				
			||||||
 | 
					        if (bits(s, 1)) {
 | 
				
			||||||
 | 
					            /* get length */
 | 
				
			||||||
 | 
					            symbol = decode(s, &lencode);
 | 
				
			||||||
 | 
					            len = base[symbol] + bits(s, extra[symbol]);
 | 
				
			||||||
 | 
					            if (len == 519) break;              /* end code */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* get distance */
 | 
				
			||||||
 | 
					            symbol = len == 2 ? 2 : dict;
 | 
				
			||||||
 | 
					            dist = decode(s, &distcode) << symbol;
 | 
				
			||||||
 | 
					            dist += bits(s, symbol);
 | 
				
			||||||
 | 
					            dist++;
 | 
				
			||||||
 | 
					            if (s->first && dist > s->next)
 | 
				
			||||||
 | 
					                return -3;              /* distance too far back */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* copy length bytes from distance bytes back */
 | 
				
			||||||
 | 
					            do {
 | 
				
			||||||
 | 
					                to = s->out + s->next;
 | 
				
			||||||
 | 
					                from = to - dist;
 | 
				
			||||||
 | 
					                copy = MAXWIN;
 | 
				
			||||||
 | 
					                if (s->next < dist) {
 | 
				
			||||||
 | 
					                    from += copy;
 | 
				
			||||||
 | 
					                    copy = dist;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                copy -= s->next;
 | 
				
			||||||
 | 
					                if (copy > len) copy = len;
 | 
				
			||||||
 | 
					                len -= copy;
 | 
				
			||||||
 | 
					                s->next += copy;
 | 
				
			||||||
 | 
					                do {
 | 
				
			||||||
 | 
					                    *to++ = *from++;
 | 
				
			||||||
 | 
					                } while (--copy);
 | 
				
			||||||
 | 
					                if (s->next == MAXWIN) {
 | 
				
			||||||
 | 
					                    if (s->outfun(s->outhow, s->out, s->next)) return 1;
 | 
				
			||||||
 | 
					                    s->next = 0;
 | 
				
			||||||
 | 
					                    s->first = 0;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            } while (len != 0);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        else {
 | 
				
			||||||
 | 
					            /* get literal and write it */
 | 
				
			||||||
 | 
					            symbol = lit ? decode(s, &litcode) : bits(s, 8);
 | 
				
			||||||
 | 
					            s->out[s->next++] = symbol;
 | 
				
			||||||
 | 
					            if (s->next == MAXWIN) {
 | 
				
			||||||
 | 
					                if (s->outfun(s->outhow, s->out, s->next)) return 1;
 | 
				
			||||||
 | 
					                s->next = 0;
 | 
				
			||||||
 | 
					                s->first = 0;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    } while (1);
 | 
				
			||||||
 | 
					    return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* See comments in blast.h */
 | 
				
			||||||
 | 
					int blast(blast_in infun, void *inhow, blast_out outfun, void *outhow,
 | 
				
			||||||
 | 
					          unsigned *left, unsigned char **in)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    struct state s;             /* input/output state */
 | 
				
			||||||
 | 
					    int err;                    /* return value */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* initialize input state */
 | 
				
			||||||
 | 
					    s.infun = infun;
 | 
				
			||||||
 | 
					    s.inhow = inhow;
 | 
				
			||||||
 | 
					    if (left != NULL && *left) {
 | 
				
			||||||
 | 
					        s.left = *left;
 | 
				
			||||||
 | 
					        s.in = *in;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        s.left = 0;
 | 
				
			||||||
 | 
					    s.bitbuf = 0;
 | 
				
			||||||
 | 
					    s.bitcnt = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* initialize output state */
 | 
				
			||||||
 | 
					    s.outfun = outfun;
 | 
				
			||||||
 | 
					    s.outhow = outhow;
 | 
				
			||||||
 | 
					    s.next = 0;
 | 
				
			||||||
 | 
					    s.first = 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* return if bits() or decode() tries to read past available input */
 | 
				
			||||||
 | 
					    if (setjmp(s.env) != 0)             /* if came back here via longjmp(), */
 | 
				
			||||||
 | 
					        err = 2;                        /*  then skip decomp(), return error */
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        err = decomp(&s);               /* decompress */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* return unused input */
 | 
				
			||||||
 | 
					    if (left != NULL)
 | 
				
			||||||
 | 
					        *left = s.left;
 | 
				
			||||||
 | 
					    if (in != NULL)
 | 
				
			||||||
 | 
					        *in = s.left ? s.in : NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* write any leftover output and update the error code if needed */
 | 
				
			||||||
 | 
					    if (err != 1 && s.next && s.outfun(s.outhow, s.out, s.next) && err == 0)
 | 
				
			||||||
 | 
					        err = 1;
 | 
				
			||||||
 | 
					    return err;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef TEST
 | 
				
			||||||
 | 
					/* Example of how to use blast() */
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include <stdlib.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define CHUNK 16384
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					local unsigned inf(void *how, unsigned char **buf)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    static unsigned char hold[CHUNK];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    *buf = hold;
 | 
				
			||||||
 | 
					    return fread(hold, 1, CHUNK, (FILE *)how);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					local int outf(void *how, unsigned char *buf, unsigned len)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    return fwrite(buf, 1, len, (FILE *)how) != len;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Decompress a PKWare Compression Library stream from stdin to stdout */
 | 
				
			||||||
 | 
					int main(void)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int ret;
 | 
				
			||||||
 | 
					    unsigned left;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* decompress to stdout */
 | 
				
			||||||
 | 
					    left = 0;
 | 
				
			||||||
 | 
					    ret = blast(inf, stdin, outf, stdout, &left, NULL);
 | 
				
			||||||
 | 
					    if (ret != 0)
 | 
				
			||||||
 | 
					        fprintf(stderr, "blast error: %d\n", ret);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* count any leftover bytes */
 | 
				
			||||||
 | 
					    while (getchar() != EOF)
 | 
				
			||||||
 | 
					        left++;
 | 
				
			||||||
 | 
					    if (left)
 | 
				
			||||||
 | 
					        fprintf(stderr, "blast warning: %u unused bytes of input\n", left);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* return blast() error code */
 | 
				
			||||||
 | 
					    return ret;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,83 @@
 | 
				
			||||||
 | 
					/* blast.h -- interface for blast.c
 | 
				
			||||||
 | 
					  Copyright (C) 2003, 2012, 2013 Mark Adler
 | 
				
			||||||
 | 
					  version 1.3, 24 Aug 2013
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  This software is provided 'as-is', without any express or implied
 | 
				
			||||||
 | 
					  warranty.  In no event will the author be held liable for any damages
 | 
				
			||||||
 | 
					  arising from the use of this software.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Permission is granted to anyone to use this software for any purpose,
 | 
				
			||||||
 | 
					  including commercial applications, and to alter it and redistribute it
 | 
				
			||||||
 | 
					  freely, subject to the following restrictions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  1. The origin of this software must not be misrepresented; you must not
 | 
				
			||||||
 | 
					     claim that you wrote the original software. If you use this software
 | 
				
			||||||
 | 
					     in a product, an acknowledgment in the product documentation would be
 | 
				
			||||||
 | 
					     appreciated but is not required.
 | 
				
			||||||
 | 
					  2. Altered source versions must be plainly marked as such, and must not be
 | 
				
			||||||
 | 
					     misrepresented as being the original software.
 | 
				
			||||||
 | 
					  3. This notice may not be removed or altered from any source distribution.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Mark Adler    madler@alumni.caltech.edu
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * blast() decompresses the PKWare Data Compression Library (DCL) compressed
 | 
				
			||||||
 | 
					 * format.  It provides the same functionality as the explode() function in
 | 
				
			||||||
 | 
					 * that library.  (Note: PKWare overused the "implode" verb, and the format
 | 
				
			||||||
 | 
					 * used by their library implode() function is completely different and
 | 
				
			||||||
 | 
					 * incompatible with the implode compression method supported by PKZIP.)
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * The binary mode for stdio functions should be used to assure that the
 | 
				
			||||||
 | 
					 * compressed data is not corrupted when read or written.  For example:
 | 
				
			||||||
 | 
					 * fopen(..., "rb") and fopen(..., "wb").
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					typedef unsigned (*blast_in)(void *how, unsigned char **buf);
 | 
				
			||||||
 | 
					typedef int (*blast_out)(void *how, unsigned char *buf, unsigned len);
 | 
				
			||||||
 | 
					/* Definitions for input/output functions passed to blast().  See below for
 | 
				
			||||||
 | 
					 * what the provided functions need to do.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int blast(blast_in infun, void *inhow, blast_out outfun, void *outhow,
 | 
				
			||||||
 | 
					          unsigned *left, unsigned char **in);
 | 
				
			||||||
 | 
					/* Decompress input to output using the provided infun() and outfun() calls.
 | 
				
			||||||
 | 
					 * On success, the return value of blast() is zero.  If there is an error in
 | 
				
			||||||
 | 
					 * the source data, i.e. it is not in the proper format, then a negative value
 | 
				
			||||||
 | 
					 * is returned.  If there is not enough input available or there is not enough
 | 
				
			||||||
 | 
					 * output space, then a positive error is returned.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * The input function is invoked: len = infun(how, &buf), where buf is set by
 | 
				
			||||||
 | 
					 * infun() to point to the input buffer, and infun() returns the number of
 | 
				
			||||||
 | 
					 * available bytes there.  If infun() returns zero, then blast() returns with
 | 
				
			||||||
 | 
					 * an input error.  (blast() only asks for input if it needs it.)  inhow is for
 | 
				
			||||||
 | 
					 * use by the application to pass an input descriptor to infun(), if desired.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * If left and in are not NULL and *left is not zero when blast() is called,
 | 
				
			||||||
 | 
					 * then the *left bytes at *in are consumed for input before infun() is used.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * The output function is invoked: err = outfun(how, buf, len), where the bytes
 | 
				
			||||||
 | 
					 * to be written are buf[0..len-1].  If err is not zero, then blast() returns
 | 
				
			||||||
 | 
					 * with an output error.  outfun() is always called with len <= 4096.  outhow
 | 
				
			||||||
 | 
					 * is for use by the application to pass an output descriptor to outfun(), if
 | 
				
			||||||
 | 
					 * desired.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * If there is any unused input, *left is set to the number of bytes that were
 | 
				
			||||||
 | 
					 * read and *in points to them.  Otherwise *left is set to zero and *in is set
 | 
				
			||||||
 | 
					 * to NULL.  If left or in are NULL, then they are not set.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * The return codes are:
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 *   2:  ran out of input before completing decompression
 | 
				
			||||||
 | 
					 *   1:  output error before completing decompression
 | 
				
			||||||
 | 
					 *   0:  successful decompression
 | 
				
			||||||
 | 
					 *  -1:  literal flag not zero or one
 | 
				
			||||||
 | 
					 *  -2:  dictionary size not in 4..6
 | 
				
			||||||
 | 
					 *  -3:  distance is too far back
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * At the bottom of blast.c is an example program that uses blast() that can be
 | 
				
			||||||
 | 
					 * compiled to produce a command-line decompression filter by defining TEST.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
											
												Binary file not shown.
											
										
									
								| 
						 | 
					@ -0,0 +1 @@
 | 
				
			||||||
 | 
					AIAIAIAIAIAIA
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,557 @@
 | 
				
			||||||
 | 
					{*******************************************************}
 | 
				
			||||||
 | 
					{                                                       }
 | 
				
			||||||
 | 
					{       Borland Delphi Supplemental Components          }
 | 
				
			||||||
 | 
					{       ZLIB Data Compression Interface Unit            }
 | 
				
			||||||
 | 
					{                                                       }
 | 
				
			||||||
 | 
					{       Copyright (c) 1997,99 Borland Corporation       }
 | 
				
			||||||
 | 
					{                                                       }
 | 
				
			||||||
 | 
					{*******************************************************}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{ Updated for zlib 1.2.x by Cosmin Truta <cosmint@cs.ubbcluj.ro> }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					unit ZLib;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					interface
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					uses SysUtils, Classes;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type
 | 
				
			||||||
 | 
					  TAlloc = function (AppData: Pointer; Items, Size: Integer): Pointer; cdecl;
 | 
				
			||||||
 | 
					  TFree = procedure (AppData, Block: Pointer); cdecl;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // Internal structure.  Ignore.
 | 
				
			||||||
 | 
					  TZStreamRec = packed record
 | 
				
			||||||
 | 
					    next_in: PChar;       // next input byte
 | 
				
			||||||
 | 
					    avail_in: Integer;    // number of bytes available at next_in
 | 
				
			||||||
 | 
					    total_in: Longint;    // total nb of input bytes read so far
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    next_out: PChar;      // next output byte should be put here
 | 
				
			||||||
 | 
					    avail_out: Integer;   // remaining free space at next_out
 | 
				
			||||||
 | 
					    total_out: Longint;   // total nb of bytes output so far
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    msg: PChar;           // last error message, NULL if no error
 | 
				
			||||||
 | 
					    internal: Pointer;    // not visible by applications
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    zalloc: TAlloc;       // used to allocate the internal state
 | 
				
			||||||
 | 
					    zfree: TFree;         // used to free the internal state
 | 
				
			||||||
 | 
					    AppData: Pointer;     // private data object passed to zalloc and zfree
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    data_type: Integer;   // best guess about the data type: ascii or binary
 | 
				
			||||||
 | 
					    adler: Longint;       // adler32 value of the uncompressed data
 | 
				
			||||||
 | 
					    reserved: Longint;    // reserved for future use
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // Abstract ancestor class
 | 
				
			||||||
 | 
					  TCustomZlibStream = class(TStream)
 | 
				
			||||||
 | 
					  private
 | 
				
			||||||
 | 
					    FStrm: TStream;
 | 
				
			||||||
 | 
					    FStrmPos: Integer;
 | 
				
			||||||
 | 
					    FOnProgress: TNotifyEvent;
 | 
				
			||||||
 | 
					    FZRec: TZStreamRec;
 | 
				
			||||||
 | 
					    FBuffer: array [Word] of Char;
 | 
				
			||||||
 | 
					  protected
 | 
				
			||||||
 | 
					    procedure Progress(Sender: TObject); dynamic;
 | 
				
			||||||
 | 
					    property OnProgress: TNotifyEvent read FOnProgress write FOnProgress;
 | 
				
			||||||
 | 
					    constructor Create(Strm: TStream);
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{ TCompressionStream compresses data on the fly as data is written to it, and
 | 
				
			||||||
 | 
					  stores the compressed data to another stream.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  TCompressionStream is write-only and strictly sequential. Reading from the
 | 
				
			||||||
 | 
					  stream will raise an exception. Using Seek to move the stream pointer
 | 
				
			||||||
 | 
					  will raise an exception.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Output data is cached internally, written to the output stream only when
 | 
				
			||||||
 | 
					  the internal output buffer is full.  All pending output data is flushed
 | 
				
			||||||
 | 
					  when the stream is destroyed.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  The Position property returns the number of uncompressed bytes of
 | 
				
			||||||
 | 
					  data that have been written to the stream so far.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  CompressionRate returns the on-the-fly percentage by which the original
 | 
				
			||||||
 | 
					  data has been compressed:  (1 - (CompressedBytes / UncompressedBytes)) * 100
 | 
				
			||||||
 | 
					  If raw data size = 100 and compressed data size = 25, the CompressionRate
 | 
				
			||||||
 | 
					  is 75%
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  The OnProgress event is called each time the output buffer is filled and
 | 
				
			||||||
 | 
					  written to the output stream.  This is useful for updating a progress
 | 
				
			||||||
 | 
					  indicator when you are writing a large chunk of data to the compression
 | 
				
			||||||
 | 
					  stream in a single call.}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  TCompressionLevel = (clNone, clFastest, clDefault, clMax);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  TCompressionStream = class(TCustomZlibStream)
 | 
				
			||||||
 | 
					  private
 | 
				
			||||||
 | 
					    function GetCompressionRate: Single;
 | 
				
			||||||
 | 
					  public
 | 
				
			||||||
 | 
					    constructor Create(CompressionLevel: TCompressionLevel; Dest: TStream);
 | 
				
			||||||
 | 
					    destructor Destroy; override;
 | 
				
			||||||
 | 
					    function Read(var Buffer; Count: Longint): Longint; override;
 | 
				
			||||||
 | 
					    function Write(const Buffer; Count: Longint): Longint; override;
 | 
				
			||||||
 | 
					    function Seek(Offset: Longint; Origin: Word): Longint; override;
 | 
				
			||||||
 | 
					    property CompressionRate: Single read GetCompressionRate;
 | 
				
			||||||
 | 
					    property OnProgress;
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{ TDecompressionStream decompresses data on the fly as data is read from it.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Compressed data comes from a separate source stream.  TDecompressionStream
 | 
				
			||||||
 | 
					  is read-only and unidirectional; you can seek forward in the stream, but not
 | 
				
			||||||
 | 
					  backwards.  The special case of setting the stream position to zero is
 | 
				
			||||||
 | 
					  allowed.  Seeking forward decompresses data until the requested position in
 | 
				
			||||||
 | 
					  the uncompressed data has been reached.  Seeking backwards, seeking relative
 | 
				
			||||||
 | 
					  to the end of the stream, requesting the size of the stream, and writing to
 | 
				
			||||||
 | 
					  the stream will raise an exception.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  The Position property returns the number of bytes of uncompressed data that
 | 
				
			||||||
 | 
					  have been read from the stream so far.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  The OnProgress event is called each time the internal input buffer of
 | 
				
			||||||
 | 
					  compressed data is exhausted and the next block is read from the input stream.
 | 
				
			||||||
 | 
					  This is useful for updating a progress indicator when you are reading a
 | 
				
			||||||
 | 
					  large chunk of data from the decompression stream in a single call.}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  TDecompressionStream = class(TCustomZlibStream)
 | 
				
			||||||
 | 
					  public
 | 
				
			||||||
 | 
					    constructor Create(Source: TStream);
 | 
				
			||||||
 | 
					    destructor Destroy; override;
 | 
				
			||||||
 | 
					    function Read(var Buffer; Count: Longint): Longint; override;
 | 
				
			||||||
 | 
					    function Write(const Buffer; Count: Longint): Longint; override;
 | 
				
			||||||
 | 
					    function Seek(Offset: Longint; Origin: Word): Longint; override;
 | 
				
			||||||
 | 
					    property OnProgress;
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{ CompressBuf compresses data, buffer to buffer, in one call.
 | 
				
			||||||
 | 
					   In: InBuf = ptr to compressed data
 | 
				
			||||||
 | 
					       InBytes = number of bytes in InBuf
 | 
				
			||||||
 | 
					  Out: OutBuf = ptr to newly allocated buffer containing decompressed data
 | 
				
			||||||
 | 
					       OutBytes = number of bytes in OutBuf   }
 | 
				
			||||||
 | 
					procedure CompressBuf(const InBuf: Pointer; InBytes: Integer;
 | 
				
			||||||
 | 
					                      out OutBuf: Pointer; out OutBytes: Integer);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{ DecompressBuf decompresses data, buffer to buffer, in one call.
 | 
				
			||||||
 | 
					   In: InBuf = ptr to compressed data
 | 
				
			||||||
 | 
					       InBytes = number of bytes in InBuf
 | 
				
			||||||
 | 
					       OutEstimate = zero, or est. size of the decompressed data
 | 
				
			||||||
 | 
					  Out: OutBuf = ptr to newly allocated buffer containing decompressed data
 | 
				
			||||||
 | 
					       OutBytes = number of bytes in OutBuf   }
 | 
				
			||||||
 | 
					procedure DecompressBuf(const InBuf: Pointer; InBytes: Integer;
 | 
				
			||||||
 | 
					 OutEstimate: Integer; out OutBuf: Pointer; out OutBytes: Integer);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{ DecompressToUserBuf decompresses data, buffer to buffer, in one call.
 | 
				
			||||||
 | 
					   In: InBuf = ptr to compressed data
 | 
				
			||||||
 | 
					       InBytes = number of bytes in InBuf
 | 
				
			||||||
 | 
					  Out: OutBuf = ptr to user-allocated buffer to contain decompressed data
 | 
				
			||||||
 | 
					       BufSize = number of bytes in OutBuf   }
 | 
				
			||||||
 | 
					procedure DecompressToUserBuf(const InBuf: Pointer; InBytes: Integer;
 | 
				
			||||||
 | 
					  const OutBuf: Pointer; BufSize: Integer);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const
 | 
				
			||||||
 | 
					  zlib_version = '1.3.1';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type
 | 
				
			||||||
 | 
					  EZlibError = class(Exception);
 | 
				
			||||||
 | 
					  ECompressionError = class(EZlibError);
 | 
				
			||||||
 | 
					  EDecompressionError = class(EZlibError);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					implementation
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					uses ZLibConst;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const
 | 
				
			||||||
 | 
					  Z_NO_FLUSH      = 0;
 | 
				
			||||||
 | 
					  Z_PARTIAL_FLUSH = 1;
 | 
				
			||||||
 | 
					  Z_SYNC_FLUSH    = 2;
 | 
				
			||||||
 | 
					  Z_FULL_FLUSH    = 3;
 | 
				
			||||||
 | 
					  Z_FINISH        = 4;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Z_OK            = 0;
 | 
				
			||||||
 | 
					  Z_STREAM_END    = 1;
 | 
				
			||||||
 | 
					  Z_NEED_DICT     = 2;
 | 
				
			||||||
 | 
					  Z_ERRNO         = (-1);
 | 
				
			||||||
 | 
					  Z_STREAM_ERROR  = (-2);
 | 
				
			||||||
 | 
					  Z_DATA_ERROR    = (-3);
 | 
				
			||||||
 | 
					  Z_MEM_ERROR     = (-4);
 | 
				
			||||||
 | 
					  Z_BUF_ERROR     = (-5);
 | 
				
			||||||
 | 
					  Z_VERSION_ERROR = (-6);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Z_NO_COMPRESSION       =   0;
 | 
				
			||||||
 | 
					  Z_BEST_SPEED           =   1;
 | 
				
			||||||
 | 
					  Z_BEST_COMPRESSION     =   9;
 | 
				
			||||||
 | 
					  Z_DEFAULT_COMPRESSION  = (-1);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Z_FILTERED            = 1;
 | 
				
			||||||
 | 
					  Z_HUFFMAN_ONLY        = 2;
 | 
				
			||||||
 | 
					  Z_RLE                 = 3;
 | 
				
			||||||
 | 
					  Z_DEFAULT_STRATEGY    = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Z_BINARY   = 0;
 | 
				
			||||||
 | 
					  Z_ASCII    = 1;
 | 
				
			||||||
 | 
					  Z_UNKNOWN  = 2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  Z_DEFLATED = 8;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{$L adler32.obj}
 | 
				
			||||||
 | 
					{$L compress.obj}
 | 
				
			||||||
 | 
					{$L crc32.obj}
 | 
				
			||||||
 | 
					{$L deflate.obj}
 | 
				
			||||||
 | 
					{$L infback.obj}
 | 
				
			||||||
 | 
					{$L inffast.obj}
 | 
				
			||||||
 | 
					{$L inflate.obj}
 | 
				
			||||||
 | 
					{$L inftrees.obj}
 | 
				
			||||||
 | 
					{$L trees.obj}
 | 
				
			||||||
 | 
					{$L uncompr.obj}
 | 
				
			||||||
 | 
					{$L zutil.obj}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure adler32; external;
 | 
				
			||||||
 | 
					procedure compressBound; external;
 | 
				
			||||||
 | 
					procedure crc32; external;
 | 
				
			||||||
 | 
					procedure deflateInit2_; external;
 | 
				
			||||||
 | 
					procedure deflateParams; external;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function _malloc(Size: Integer): Pointer; cdecl;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  Result := AllocMem(Size);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure _free(Block: Pointer); cdecl;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FreeMem(Block);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure _memset(P: Pointer; B: Byte; count: Integer); cdecl;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FillChar(P^, count, B);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure _memcpy(dest, source: Pointer; count: Integer); cdecl;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  Move(source^, dest^, count);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// deflate compresses data
 | 
				
			||||||
 | 
					function deflateInit_(var strm: TZStreamRec; level: Integer; version: PChar;
 | 
				
			||||||
 | 
					  recsize: Integer): Integer; external;
 | 
				
			||||||
 | 
					function deflate(var strm: TZStreamRec; flush: Integer): Integer; external;
 | 
				
			||||||
 | 
					function deflateEnd(var strm: TZStreamRec): Integer; external;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// inflate decompresses data
 | 
				
			||||||
 | 
					function inflateInit_(var strm: TZStreamRec; version: PChar;
 | 
				
			||||||
 | 
					  recsize: Integer): Integer; external;
 | 
				
			||||||
 | 
					function inflate(var strm: TZStreamRec; flush: Integer): Integer; external;
 | 
				
			||||||
 | 
					function inflateEnd(var strm: TZStreamRec): Integer; external;
 | 
				
			||||||
 | 
					function inflateReset(var strm: TZStreamRec): Integer; external;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function zlibAllocMem(AppData: Pointer; Items, Size: Integer): Pointer; cdecl;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					//  GetMem(Result, Items*Size);
 | 
				
			||||||
 | 
					  Result := AllocMem(Items * Size);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure zlibFreeMem(AppData, Block: Pointer); cdecl;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FreeMem(Block);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					{function zlibCheck(code: Integer): Integer;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  Result := code;
 | 
				
			||||||
 | 
					  if code < 0 then
 | 
				
			||||||
 | 
					    raise EZlibError.Create('error');    //!!
 | 
				
			||||||
 | 
					end;}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function CCheck(code: Integer): Integer;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  Result := code;
 | 
				
			||||||
 | 
					  if code < 0 then
 | 
				
			||||||
 | 
					    raise ECompressionError.Create('error'); //!!
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function DCheck(code: Integer): Integer;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  Result := code;
 | 
				
			||||||
 | 
					  if code < 0 then
 | 
				
			||||||
 | 
					    raise EDecompressionError.Create('error');  //!!
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure CompressBuf(const InBuf: Pointer; InBytes: Integer;
 | 
				
			||||||
 | 
					                      out OutBuf: Pointer; out OutBytes: Integer);
 | 
				
			||||||
 | 
					var
 | 
				
			||||||
 | 
					  strm: TZStreamRec;
 | 
				
			||||||
 | 
					  P: Pointer;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FillChar(strm, sizeof(strm), 0);
 | 
				
			||||||
 | 
					  strm.zalloc := zlibAllocMem;
 | 
				
			||||||
 | 
					  strm.zfree := zlibFreeMem;
 | 
				
			||||||
 | 
					  OutBytes := ((InBytes + (InBytes div 10) + 12) + 255) and not 255;
 | 
				
			||||||
 | 
					  GetMem(OutBuf, OutBytes);
 | 
				
			||||||
 | 
					  try
 | 
				
			||||||
 | 
					    strm.next_in := InBuf;
 | 
				
			||||||
 | 
					    strm.avail_in := InBytes;
 | 
				
			||||||
 | 
					    strm.next_out := OutBuf;
 | 
				
			||||||
 | 
					    strm.avail_out := OutBytes;
 | 
				
			||||||
 | 
					    CCheck(deflateInit_(strm, Z_BEST_COMPRESSION, zlib_version, sizeof(strm)));
 | 
				
			||||||
 | 
					    try
 | 
				
			||||||
 | 
					      while CCheck(deflate(strm, Z_FINISH)) <> Z_STREAM_END do
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					        P := OutBuf;
 | 
				
			||||||
 | 
					        Inc(OutBytes, 256);
 | 
				
			||||||
 | 
					        ReallocMem(OutBuf, OutBytes);
 | 
				
			||||||
 | 
					        strm.next_out := PChar(Integer(OutBuf) + (Integer(strm.next_out) - Integer(P)));
 | 
				
			||||||
 | 
					        strm.avail_out := 256;
 | 
				
			||||||
 | 
					      end;
 | 
				
			||||||
 | 
					    finally
 | 
				
			||||||
 | 
					      CCheck(deflateEnd(strm));
 | 
				
			||||||
 | 
					    end;
 | 
				
			||||||
 | 
					    ReallocMem(OutBuf, strm.total_out);
 | 
				
			||||||
 | 
					    OutBytes := strm.total_out;
 | 
				
			||||||
 | 
					  except
 | 
				
			||||||
 | 
					    FreeMem(OutBuf);
 | 
				
			||||||
 | 
					    raise
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure DecompressBuf(const InBuf: Pointer; InBytes: Integer;
 | 
				
			||||||
 | 
					  OutEstimate: Integer; out OutBuf: Pointer; out OutBytes: Integer);
 | 
				
			||||||
 | 
					var
 | 
				
			||||||
 | 
					  strm: TZStreamRec;
 | 
				
			||||||
 | 
					  P: Pointer;
 | 
				
			||||||
 | 
					  BufInc: Integer;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FillChar(strm, sizeof(strm), 0);
 | 
				
			||||||
 | 
					  strm.zalloc := zlibAllocMem;
 | 
				
			||||||
 | 
					  strm.zfree := zlibFreeMem;
 | 
				
			||||||
 | 
					  BufInc := (InBytes + 255) and not 255;
 | 
				
			||||||
 | 
					  if OutEstimate = 0 then
 | 
				
			||||||
 | 
					    OutBytes := BufInc
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    OutBytes := OutEstimate;
 | 
				
			||||||
 | 
					  GetMem(OutBuf, OutBytes);
 | 
				
			||||||
 | 
					  try
 | 
				
			||||||
 | 
					    strm.next_in := InBuf;
 | 
				
			||||||
 | 
					    strm.avail_in := InBytes;
 | 
				
			||||||
 | 
					    strm.next_out := OutBuf;
 | 
				
			||||||
 | 
					    strm.avail_out := OutBytes;
 | 
				
			||||||
 | 
					    DCheck(inflateInit_(strm, zlib_version, sizeof(strm)));
 | 
				
			||||||
 | 
					    try
 | 
				
			||||||
 | 
					      while DCheck(inflate(strm, Z_NO_FLUSH)) <> Z_STREAM_END do
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					        P := OutBuf;
 | 
				
			||||||
 | 
					        Inc(OutBytes, BufInc);
 | 
				
			||||||
 | 
					        ReallocMem(OutBuf, OutBytes);
 | 
				
			||||||
 | 
					        strm.next_out := PChar(Integer(OutBuf) + (Integer(strm.next_out) - Integer(P)));
 | 
				
			||||||
 | 
					        strm.avail_out := BufInc;
 | 
				
			||||||
 | 
					      end;
 | 
				
			||||||
 | 
					    finally
 | 
				
			||||||
 | 
					      DCheck(inflateEnd(strm));
 | 
				
			||||||
 | 
					    end;
 | 
				
			||||||
 | 
					    ReallocMem(OutBuf, strm.total_out);
 | 
				
			||||||
 | 
					    OutBytes := strm.total_out;
 | 
				
			||||||
 | 
					  except
 | 
				
			||||||
 | 
					    FreeMem(OutBuf);
 | 
				
			||||||
 | 
					    raise
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure DecompressToUserBuf(const InBuf: Pointer; InBytes: Integer;
 | 
				
			||||||
 | 
					  const OutBuf: Pointer; BufSize: Integer);
 | 
				
			||||||
 | 
					var
 | 
				
			||||||
 | 
					  strm: TZStreamRec;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FillChar(strm, sizeof(strm), 0);
 | 
				
			||||||
 | 
					  strm.zalloc := zlibAllocMem;
 | 
				
			||||||
 | 
					  strm.zfree := zlibFreeMem;
 | 
				
			||||||
 | 
					  strm.next_in := InBuf;
 | 
				
			||||||
 | 
					  strm.avail_in := InBytes;
 | 
				
			||||||
 | 
					  strm.next_out := OutBuf;
 | 
				
			||||||
 | 
					  strm.avail_out := BufSize;
 | 
				
			||||||
 | 
					  DCheck(inflateInit_(strm, zlib_version, sizeof(strm)));
 | 
				
			||||||
 | 
					  try
 | 
				
			||||||
 | 
					    if DCheck(inflate(strm, Z_FINISH)) <> Z_STREAM_END then
 | 
				
			||||||
 | 
					      raise EZlibError.CreateRes(@sTargetBufferTooSmall);
 | 
				
			||||||
 | 
					  finally
 | 
				
			||||||
 | 
					    DCheck(inflateEnd(strm));
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TCustomZlibStream
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					constructor TCustomZLibStream.Create(Strm: TStream);
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  inherited Create;
 | 
				
			||||||
 | 
					  FStrm := Strm;
 | 
				
			||||||
 | 
					  FStrmPos := Strm.Position;
 | 
				
			||||||
 | 
					  FZRec.zalloc := zlibAllocMem;
 | 
				
			||||||
 | 
					  FZRec.zfree := zlibFreeMem;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					procedure TCustomZLibStream.Progress(Sender: TObject);
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  if Assigned(FOnProgress) then FOnProgress(Sender);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TCompressionStream
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					constructor TCompressionStream.Create(CompressionLevel: TCompressionLevel;
 | 
				
			||||||
 | 
					  Dest: TStream);
 | 
				
			||||||
 | 
					const
 | 
				
			||||||
 | 
					  Levels: array [TCompressionLevel] of ShortInt =
 | 
				
			||||||
 | 
					    (Z_NO_COMPRESSION, Z_BEST_SPEED, Z_DEFAULT_COMPRESSION, Z_BEST_COMPRESSION);
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  inherited Create(Dest);
 | 
				
			||||||
 | 
					  FZRec.next_out := FBuffer;
 | 
				
			||||||
 | 
					  FZRec.avail_out := sizeof(FBuffer);
 | 
				
			||||||
 | 
					  CCheck(deflateInit_(FZRec, Levels[CompressionLevel], zlib_version, sizeof(FZRec)));
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					destructor TCompressionStream.Destroy;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FZRec.next_in := nil;
 | 
				
			||||||
 | 
					  FZRec.avail_in := 0;
 | 
				
			||||||
 | 
					  try
 | 
				
			||||||
 | 
					    if FStrm.Position <> FStrmPos then FStrm.Position := FStrmPos;
 | 
				
			||||||
 | 
					    while (CCheck(deflate(FZRec, Z_FINISH)) <> Z_STREAM_END)
 | 
				
			||||||
 | 
					      and (FZRec.avail_out = 0) do
 | 
				
			||||||
 | 
					    begin
 | 
				
			||||||
 | 
					      FStrm.WriteBuffer(FBuffer, sizeof(FBuffer));
 | 
				
			||||||
 | 
					      FZRec.next_out := FBuffer;
 | 
				
			||||||
 | 
					      FZRec.avail_out := sizeof(FBuffer);
 | 
				
			||||||
 | 
					    end;
 | 
				
			||||||
 | 
					    if FZRec.avail_out < sizeof(FBuffer) then
 | 
				
			||||||
 | 
					      FStrm.WriteBuffer(FBuffer, sizeof(FBuffer) - FZRec.avail_out);
 | 
				
			||||||
 | 
					  finally
 | 
				
			||||||
 | 
					    deflateEnd(FZRec);
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					  inherited Destroy;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function TCompressionStream.Read(var Buffer; Count: Longint): Longint;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  raise ECompressionError.CreateRes(@sInvalidStreamOp);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function TCompressionStream.Write(const Buffer; Count: Longint): Longint;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FZRec.next_in := @Buffer;
 | 
				
			||||||
 | 
					  FZRec.avail_in := Count;
 | 
				
			||||||
 | 
					  if FStrm.Position <> FStrmPos then FStrm.Position := FStrmPos;
 | 
				
			||||||
 | 
					  while (FZRec.avail_in > 0) do
 | 
				
			||||||
 | 
					  begin
 | 
				
			||||||
 | 
					    CCheck(deflate(FZRec, 0));
 | 
				
			||||||
 | 
					    if FZRec.avail_out = 0 then
 | 
				
			||||||
 | 
					    begin
 | 
				
			||||||
 | 
					      FStrm.WriteBuffer(FBuffer, sizeof(FBuffer));
 | 
				
			||||||
 | 
					      FZRec.next_out := FBuffer;
 | 
				
			||||||
 | 
					      FZRec.avail_out := sizeof(FBuffer);
 | 
				
			||||||
 | 
					      FStrmPos := FStrm.Position;
 | 
				
			||||||
 | 
					      Progress(Self);
 | 
				
			||||||
 | 
					    end;
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					  Result := Count;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function TCompressionStream.Seek(Offset: Longint; Origin: Word): Longint;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  if (Offset = 0) and (Origin = soFromCurrent) then
 | 
				
			||||||
 | 
					    Result := FZRec.total_in
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    raise ECompressionError.CreateRes(@sInvalidStreamOp);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function TCompressionStream.GetCompressionRate: Single;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  if FZRec.total_in = 0 then
 | 
				
			||||||
 | 
					    Result := 0
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    Result := (1.0 - (FZRec.total_out / FZRec.total_in)) * 100.0;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TDecompressionStream
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					constructor TDecompressionStream.Create(Source: TStream);
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  inherited Create(Source);
 | 
				
			||||||
 | 
					  FZRec.next_in := FBuffer;
 | 
				
			||||||
 | 
					  FZRec.avail_in := 0;
 | 
				
			||||||
 | 
					  DCheck(inflateInit_(FZRec, zlib_version, sizeof(FZRec)));
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					destructor TDecompressionStream.Destroy;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FStrm.Seek(-FZRec.avail_in, 1);
 | 
				
			||||||
 | 
					  inflateEnd(FZRec);
 | 
				
			||||||
 | 
					  inherited Destroy;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function TDecompressionStream.Read(var Buffer; Count: Longint): Longint;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  FZRec.next_out := @Buffer;
 | 
				
			||||||
 | 
					  FZRec.avail_out := Count;
 | 
				
			||||||
 | 
					  if FStrm.Position <> FStrmPos then FStrm.Position := FStrmPos;
 | 
				
			||||||
 | 
					  while (FZRec.avail_out > 0) do
 | 
				
			||||||
 | 
					  begin
 | 
				
			||||||
 | 
					    if FZRec.avail_in = 0 then
 | 
				
			||||||
 | 
					    begin
 | 
				
			||||||
 | 
					      FZRec.avail_in := FStrm.Read(FBuffer, sizeof(FBuffer));
 | 
				
			||||||
 | 
					      if FZRec.avail_in = 0 then
 | 
				
			||||||
 | 
					      begin
 | 
				
			||||||
 | 
					        Result := Count - FZRec.avail_out;
 | 
				
			||||||
 | 
					        Exit;
 | 
				
			||||||
 | 
					      end;
 | 
				
			||||||
 | 
					      FZRec.next_in := FBuffer;
 | 
				
			||||||
 | 
					      FStrmPos := FStrm.Position;
 | 
				
			||||||
 | 
					      Progress(Self);
 | 
				
			||||||
 | 
					    end;
 | 
				
			||||||
 | 
					    CCheck(inflate(FZRec, 0));
 | 
				
			||||||
 | 
					  end;
 | 
				
			||||||
 | 
					  Result := Count;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function TDecompressionStream.Write(const Buffer; Count: Longint): Longint;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  raise EDecompressionError.CreateRes(@sInvalidStreamOp);
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function TDecompressionStream.Seek(Offset: Longint; Origin: Word): Longint;
 | 
				
			||||||
 | 
					var
 | 
				
			||||||
 | 
					  I: Integer;
 | 
				
			||||||
 | 
					  Buf: array [0..4095] of Char;
 | 
				
			||||||
 | 
					begin
 | 
				
			||||||
 | 
					  if (Offset = 0) and (Origin = soFromBeginning) then
 | 
				
			||||||
 | 
					  begin
 | 
				
			||||||
 | 
					    DCheck(inflateReset(FZRec));
 | 
				
			||||||
 | 
					    FZRec.next_in := FBuffer;
 | 
				
			||||||
 | 
					    FZRec.avail_in := 0;
 | 
				
			||||||
 | 
					    FStrm.Position := 0;
 | 
				
			||||||
 | 
					    FStrmPos := 0;
 | 
				
			||||||
 | 
					  end
 | 
				
			||||||
 | 
					  else if ( (Offset >= 0) and (Origin = soFromCurrent)) or
 | 
				
			||||||
 | 
					          ( ((Offset - FZRec.total_out) > 0) and (Origin = soFromBeginning)) then
 | 
				
			||||||
 | 
					  begin
 | 
				
			||||||
 | 
					    if Origin = soFromBeginning then Dec(Offset, FZRec.total_out);
 | 
				
			||||||
 | 
					    if Offset > 0 then
 | 
				
			||||||
 | 
					    begin
 | 
				
			||||||
 | 
					      for I := 1 to Offset div sizeof(Buf) do
 | 
				
			||||||
 | 
					        ReadBuffer(Buf, sizeof(Buf));
 | 
				
			||||||
 | 
					      ReadBuffer(Buf, Offset mod sizeof(Buf));
 | 
				
			||||||
 | 
					    end;
 | 
				
			||||||
 | 
					  end
 | 
				
			||||||
 | 
					  else
 | 
				
			||||||
 | 
					    raise EDecompressionError.CreateRes(@sInvalidStreamOp);
 | 
				
			||||||
 | 
					  Result := FZRec.total_out;
 | 
				
			||||||
 | 
					end;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end.
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,11 @@
 | 
				
			||||||
 | 
					unit ZLibConst;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					interface
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					resourcestring
 | 
				
			||||||
 | 
					  sTargetBufferTooSmall = 'ZLib error: target buffer may be too small';
 | 
				
			||||||
 | 
					  sInvalidStreamOp = 'Invalid stream operation';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					implementation
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					end.
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,76 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Overview
 | 
				
			||||||
 | 
					========
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					This directory contains an update to the ZLib interface unit,
 | 
				
			||||||
 | 
					distributed by Borland as a Delphi supplemental component.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The original ZLib unit is Copyright (c) 1997,99 Borland Corp.,
 | 
				
			||||||
 | 
					and is based on zlib version 1.0.4.  There are a series of bugs
 | 
				
			||||||
 | 
					and security problems associated with that old zlib version, and
 | 
				
			||||||
 | 
					we recommend the users to update their ZLib unit.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Summary of modifications
 | 
				
			||||||
 | 
					========================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Improved makefile, adapted to zlib version 1.2.1.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Some field types from TZStreamRec are changed from Integer to
 | 
				
			||||||
 | 
					  Longint, for consistency with the zlib.h header, and for 64-bit
 | 
				
			||||||
 | 
					  readiness.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- The zlib_version constant is updated.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- The new Z_RLE strategy has its corresponding symbolic constant.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- The allocation and deallocation functions and function types
 | 
				
			||||||
 | 
					  (TAlloc, TFree, zlibAllocMem and zlibFreeMem) are now cdecl,
 | 
				
			||||||
 | 
					  and _malloc and _free are added as C RTL stubs.  As a result,
 | 
				
			||||||
 | 
					  the original C sources of zlib can be compiled out of the box,
 | 
				
			||||||
 | 
					  and linked to the ZLib unit.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Suggestions for improvements
 | 
				
			||||||
 | 
					============================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Currently, the ZLib unit provides only a limited wrapper around
 | 
				
			||||||
 | 
					the zlib library, and much of the original zlib functionality is
 | 
				
			||||||
 | 
					missing.  Handling compressed file formats like ZIP/GZIP or PNG
 | 
				
			||||||
 | 
					cannot be implemented without having this functionality.
 | 
				
			||||||
 | 
					Applications that handle these formats are either using their own,
 | 
				
			||||||
 | 
					duplicated code, or not using the ZLib unit at all.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Here are a few suggestions:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- Checksum class wrappers around adler32() and crc32(), similar
 | 
				
			||||||
 | 
					  to the Java classes that implement the java.util.zip.Checksum
 | 
				
			||||||
 | 
					  interface.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- The ability to read and write raw deflate streams, without the
 | 
				
			||||||
 | 
					  zlib stream header and trailer.  Raw deflate streams are used
 | 
				
			||||||
 | 
					  in the ZIP file format.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- The ability to read and write gzip streams, used in the GZIP
 | 
				
			||||||
 | 
					  file format, and normally produced by the gzip program.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- The ability to select a different compression strategy, useful
 | 
				
			||||||
 | 
					  to PNG and MNG image compression, and to multimedia compression
 | 
				
			||||||
 | 
					  in general.  Besides the compression level
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    TCompressionLevel = (clNone, clFastest, clDefault, clMax);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  which, in fact, could have used the 'z' prefix and avoided
 | 
				
			||||||
 | 
					  TColor-like symbols
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    TCompressionLevel = (zcNone, zcFastest, zcDefault, zcMax);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  there could be a compression strategy
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    TCompressionStrategy = (zsDefault, zsFiltered, zsHuffmanOnly, zsRle);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- ZIP and GZIP stream handling via TStreams.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					--
 | 
				
			||||||
 | 
					Cosmin Truta <cosmint@cs.ubbcluj.ro>
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,99 @@
 | 
				
			||||||
 | 
					# Makefile for zlib
 | 
				
			||||||
 | 
					# For use with Delphi and C++ Builder under Win32
 | 
				
			||||||
 | 
					# Updated for zlib 1.2.x by Cosmin Truta
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# ------------ Borland C++ ------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# This project uses the Delphi (fastcall/register) calling convention:
 | 
				
			||||||
 | 
					LOC = -DZEXPORT=__fastcall -DZEXPORTVA=__cdecl
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					CC = bcc32
 | 
				
			||||||
 | 
					LD = bcc32
 | 
				
			||||||
 | 
					AR = tlib
 | 
				
			||||||
 | 
					# do not use "-pr" in CFLAGS
 | 
				
			||||||
 | 
					CFLAGS = -a -d -k- -O2 $(LOC)
 | 
				
			||||||
 | 
					LDFLAGS =
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# variables
 | 
				
			||||||
 | 
					ZLIB_LIB = zlib.lib
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					OBJ1 = adler32.obj compress.obj crc32.obj deflate.obj gzclose.obj gzlib.obj gzread.obj
 | 
				
			||||||
 | 
					OBJ2 = gzwrite.obj infback.obj inffast.obj inflate.obj inftrees.obj trees.obj uncompr.obj zutil.obj
 | 
				
			||||||
 | 
					OBJP1 = +adler32.obj+compress.obj+crc32.obj+deflate.obj+gzclose.obj+gzlib.obj+gzread.obj
 | 
				
			||||||
 | 
					OBJP2 = +gzwrite.obj+infback.obj+inffast.obj+inflate.obj+inftrees.obj+trees.obj+uncompr.obj+zutil.obj
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# targets
 | 
				
			||||||
 | 
					all: $(ZLIB_LIB) example.exe minigzip.exe
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.c.obj:
 | 
				
			||||||
 | 
						$(CC) -c $(CFLAGS) $*.c
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					adler32.obj: adler32.c zlib.h zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					compress.obj: compress.c zlib.h zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					crc32.obj: crc32.c zlib.h zconf.h crc32.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					deflate.obj: deflate.c deflate.h zutil.h zlib.h zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzclose.obj: gzclose.c zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzlib.obj: gzlib.c zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzread.obj: gzread.c zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					gzwrite.obj: gzwrite.c zlib.h zconf.h gzguts.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					infback.obj: infback.c zutil.h zlib.h zconf.h inftrees.h inflate.h \
 | 
				
			||||||
 | 
					 inffast.h inffixed.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inffast.obj: inffast.c zutil.h zlib.h zconf.h inftrees.h inflate.h \
 | 
				
			||||||
 | 
					 inffast.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inflate.obj: inflate.c zutil.h zlib.h zconf.h inftrees.h inflate.h \
 | 
				
			||||||
 | 
					 inffast.h inffixed.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					inftrees.obj: inftrees.c zutil.h zlib.h zconf.h inftrees.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					trees.obj: trees.c zutil.h zlib.h zconf.h deflate.h trees.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					uncompr.obj: uncompr.c zlib.h zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					zutil.obj: zutil.c zutil.h zlib.h zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example.obj: test/example.c zlib.h zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip.obj: test/minigzip.c zlib.h zconf.h
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# For the sake of the old Borland make,
 | 
				
			||||||
 | 
					# the command line is cut to fit in the MS-DOS 128 byte limit:
 | 
				
			||||||
 | 
					$(ZLIB_LIB): $(OBJ1) $(OBJ2)
 | 
				
			||||||
 | 
						-del $(ZLIB_LIB)
 | 
				
			||||||
 | 
						$(AR) $(ZLIB_LIB) $(OBJP1)
 | 
				
			||||||
 | 
						$(AR) $(ZLIB_LIB) $(OBJP2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# testing
 | 
				
			||||||
 | 
					test: example.exe minigzip.exe
 | 
				
			||||||
 | 
						example
 | 
				
			||||||
 | 
						echo hello world | minigzip | minigzip -d
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					example.exe: example.obj $(ZLIB_LIB)
 | 
				
			||||||
 | 
						$(LD) $(LDFLAGS) example.obj $(ZLIB_LIB)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					minigzip.exe: minigzip.obj $(ZLIB_LIB)
 | 
				
			||||||
 | 
						$(LD) $(LDFLAGS) minigzip.obj $(ZLIB_LIB)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# cleanup
 | 
				
			||||||
 | 
					clean:
 | 
				
			||||||
 | 
						-del *.obj
 | 
				
			||||||
 | 
						-del *.exe
 | 
				
			||||||
 | 
						-del *.lib
 | 
				
			||||||
 | 
						-del *.tds
 | 
				
			||||||
 | 
						-del zlib.bak
 | 
				
			||||||
 | 
						-del foo.gz
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,33 @@
 | 
				
			||||||
 | 
					<?xml version="1.0" encoding="utf-8" ?>
 | 
				
			||||||
 | 
					<project name="DotZLib" default="build" basedir="./DotZLib">
 | 
				
			||||||
 | 
						<description>A .Net wrapper library around ZLib1.dll</description>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<property name="nunit.location" value="c:/program files/NUnit V2.1/bin" />
 | 
				
			||||||
 | 
						<property name="build.root" value="bin" />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<property name="debug" value="true" />
 | 
				
			||||||
 | 
						<property name="nunit" value="true" />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<property name="build.folder" value="${build.root}/debug/" if="${debug}" />
 | 
				
			||||||
 | 
						<property name="build.folder" value="${build.root}/release/" unless="${debug}" />
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<target name="clean" description="Remove all generated files">
 | 
				
			||||||
 | 
							<delete dir="${build.root}" failonerror="false" />
 | 
				
			||||||
 | 
						</target>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						<target name="build" description="compiles the source code">
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							<mkdir dir="${build.folder}" />
 | 
				
			||||||
 | 
							<csc target="library" output="${build.folder}DotZLib.dll" debug="${debug}">
 | 
				
			||||||
 | 
								<references basedir="${nunit.location}">
 | 
				
			||||||
 | 
									<includes if="${nunit}" name="nunit.framework.dll" />
 | 
				
			||||||
 | 
								</references>
 | 
				
			||||||
 | 
								<sources>
 | 
				
			||||||
 | 
									<includes name="*.cs" />
 | 
				
			||||||
 | 
									<excludes name="UnitTests.cs" unless="${nunit}" />
 | 
				
			||||||
 | 
								</sources>
 | 
				
			||||||
 | 
								<arg value="/d:nunit" if="${nunit}" />
 | 
				
			||||||
 | 
							</csc>
 | 
				
			||||||
 | 
						</target>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					</project>
 | 
				
			||||||
											
												Binary file not shown.
											
										
									
								| 
						 | 
					@ -0,0 +1,21 @@
 | 
				
			||||||
 | 
					Microsoft Visual Studio Solution File, Format Version 8.00
 | 
				
			||||||
 | 
					Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "DotZLib", "DotZLib\DotZLib.csproj", "{BB1EE0B1-1808-46CB-B786-949D91117FC5}"
 | 
				
			||||||
 | 
						ProjectSection(ProjectDependencies) = postProject
 | 
				
			||||||
 | 
						EndProjectSection
 | 
				
			||||||
 | 
					EndProject
 | 
				
			||||||
 | 
					Global
 | 
				
			||||||
 | 
						GlobalSection(SolutionConfiguration) = preSolution
 | 
				
			||||||
 | 
							Debug = Debug
 | 
				
			||||||
 | 
							Release = Release
 | 
				
			||||||
 | 
						EndGlobalSection
 | 
				
			||||||
 | 
						GlobalSection(ProjectConfiguration) = postSolution
 | 
				
			||||||
 | 
							{BB1EE0B1-1808-46CB-B786-949D91117FC5}.Debug.ActiveCfg = Debug|.NET
 | 
				
			||||||
 | 
							{BB1EE0B1-1808-46CB-B786-949D91117FC5}.Debug.Build.0 = Debug|.NET
 | 
				
			||||||
 | 
							{BB1EE0B1-1808-46CB-B786-949D91117FC5}.Release.ActiveCfg = Release|.NET
 | 
				
			||||||
 | 
							{BB1EE0B1-1808-46CB-B786-949D91117FC5}.Release.Build.0 = Release|.NET
 | 
				
			||||||
 | 
						EndGlobalSection
 | 
				
			||||||
 | 
						GlobalSection(ExtensibilityGlobals) = postSolution
 | 
				
			||||||
 | 
						EndGlobalSection
 | 
				
			||||||
 | 
						GlobalSection(ExtensibilityAddIns) = postSolution
 | 
				
			||||||
 | 
						EndGlobalSection
 | 
				
			||||||
 | 
					EndGlobal
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,58 @@
 | 
				
			||||||
 | 
					using System.Reflection;
 | 
				
			||||||
 | 
					using System.Runtime.CompilerServices;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// General Information about an assembly is controlled through the following
 | 
				
			||||||
 | 
					// set of attributes. Change these attribute values to modify the information
 | 
				
			||||||
 | 
					// associated with an assembly.
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					[assembly: AssemblyTitle("DotZLib")]
 | 
				
			||||||
 | 
					[assembly: AssemblyDescription(".Net bindings for ZLib compression dll 1.2.x")]
 | 
				
			||||||
 | 
					[assembly: AssemblyConfiguration("")]
 | 
				
			||||||
 | 
					[assembly: AssemblyCompany("Henrik Ravn")]
 | 
				
			||||||
 | 
					[assembly: AssemblyProduct("")]
 | 
				
			||||||
 | 
					[assembly: AssemblyCopyright("(c) 2004 by Henrik Ravn")]
 | 
				
			||||||
 | 
					[assembly: AssemblyTrademark("")]
 | 
				
			||||||
 | 
					[assembly: AssemblyCulture("")]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Version information for an assembly consists of the following four values:
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					//      Major Version
 | 
				
			||||||
 | 
					//      Minor Version
 | 
				
			||||||
 | 
					//      Build Number
 | 
				
			||||||
 | 
					//      Revision
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// You can specify all the values or you can default the Revision and Build Numbers
 | 
				
			||||||
 | 
					// by using the '*' as shown below:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					[assembly: AssemblyVersion("1.0.*")]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// In order to sign your assembly you must specify a key to use. Refer to the
 | 
				
			||||||
 | 
					// Microsoft .NET Framework documentation for more information on assembly signing.
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use the attributes below to control which key is used for signing.
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Notes:
 | 
				
			||||||
 | 
					//   (*) If no key is specified, the assembly is not signed.
 | 
				
			||||||
 | 
					//   (*) KeyName refers to a key that has been installed in the Crypto Service
 | 
				
			||||||
 | 
					//       Provider (CSP) on your machine. KeyFile refers to a file which contains
 | 
				
			||||||
 | 
					//       a key.
 | 
				
			||||||
 | 
					//   (*) If the KeyFile and the KeyName values are both specified, the
 | 
				
			||||||
 | 
					//       following processing occurs:
 | 
				
			||||||
 | 
					//       (1) If the KeyName can be found in the CSP, that key is used.
 | 
				
			||||||
 | 
					//       (2) If the KeyName does not exist and the KeyFile does exist, the key
 | 
				
			||||||
 | 
					//           in the KeyFile is installed into the CSP and used.
 | 
				
			||||||
 | 
					//   (*) In order to create a KeyFile, you can use the sn.exe (Strong Name) utility.
 | 
				
			||||||
 | 
					//       When specifying the KeyFile, the location of the KeyFile should be
 | 
				
			||||||
 | 
					//       relative to the project output directory which is
 | 
				
			||||||
 | 
					//       %Project Directory%\obj\<configuration>. For example, if your KeyFile is
 | 
				
			||||||
 | 
					//       located in the project directory, you would specify the AssemblyKeyFile
 | 
				
			||||||
 | 
					//       attribute as [assembly: AssemblyKeyFile("..\\..\\mykey.snk")]
 | 
				
			||||||
 | 
					//   (*) Delay Signing is an advanced option - see the Microsoft .NET Framework
 | 
				
			||||||
 | 
					//       documentation for more information on this.
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					[assembly: AssemblyDelaySign(false)]
 | 
				
			||||||
 | 
					[assembly: AssemblyKeyFile("")]
 | 
				
			||||||
 | 
					[assembly: AssemblyKeyName("")]
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,202 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.Runtime.InteropServices;
 | 
				
			||||||
 | 
					using System.Text;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLib
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    #region ChecksumGeneratorBase
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Implements the common functionality needed for all <see cref="ChecksumGenerator"/>s
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    /// <example></example>
 | 
				
			||||||
 | 
					    public abstract class ChecksumGeneratorBase : ChecksumGenerator
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// The value of the current checksum
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        protected uint _current;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the checksum generator base - the current checksum is
 | 
				
			||||||
 | 
					        /// set to zero
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public ChecksumGeneratorBase()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            _current = 0;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the checksum generator base with a specified value
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="initialValue">The value to set the current checksum to</param>
 | 
				
			||||||
 | 
					        public ChecksumGeneratorBase(uint initialValue)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            _current = initialValue;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Resets the current checksum to zero
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public void Reset() { _current = 0; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the current checksum value
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public uint Value { get { return _current; } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with part of an array of bytes
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The data to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">Where in <c>data</c> to start updating</param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes from <c>data</c> to use</param>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentException">The sum of offset and count is larger than the length of <c>data</c></exception>
 | 
				
			||||||
 | 
					        /// <exception cref="NullReferenceException"><c>data</c> is a null reference</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentOutOfRangeException">Offset or count is negative.</exception>
 | 
				
			||||||
 | 
					        /// <remarks>All the other <c>Update</c> methods are implemented in terms of this one.
 | 
				
			||||||
 | 
					        /// This is therefore the only method a derived class has to implement</remarks>
 | 
				
			||||||
 | 
					        public abstract void Update(byte[] data, int offset, int count);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with an array of bytes.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The data to update the checksum with</param>
 | 
				
			||||||
 | 
					        public void Update(byte[] data)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            Update(data, 0, data.Length);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with the data from a string
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The string to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <remarks>The characters in the string are converted by the UTF-8 encoding</remarks>
 | 
				
			||||||
 | 
					        public void Update(string data)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
								Update(Encoding.UTF8.GetBytes(data));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with the data from a string, using a specific encoding
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The string to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <param name="encoding">The encoding to use</param>
 | 
				
			||||||
 | 
					        public void Update(string data, Encoding encoding)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            Update(encoding.GetBytes(data));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region CRC32
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Implements a CRC32 checksum generator
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public sealed class CRC32Checksum : ChecksumGeneratorBase
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region DLL imports
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern uint crc32(uint crc, int data, uint length);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the CRC32 checksum generator
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public CRC32Checksum() : base() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the CRC32 checksum generator with a specified value
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="initialValue">The value to set the current checksum to</param>
 | 
				
			||||||
 | 
					        public CRC32Checksum(uint initialValue) : base(initialValue) {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with part of an array of bytes
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The data to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">Where in <c>data</c> to start updating</param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes from <c>data</c> to use</param>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentException">The sum of offset and count is larger than the length of <c>data</c></exception>
 | 
				
			||||||
 | 
					        /// <exception cref="NullReferenceException"><c>data</c> is a null reference</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentOutOfRangeException">Offset or count is negative.</exception>
 | 
				
			||||||
 | 
					        public override void Update(byte[] data, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (offset < 0 || count < 0) throw new ArgumentOutOfRangeException();
 | 
				
			||||||
 | 
					            if ((offset+count) > data.Length) throw new ArgumentException();
 | 
				
			||||||
 | 
					            GCHandle hData = GCHandle.Alloc(data, GCHandleType.Pinned);
 | 
				
			||||||
 | 
					            try
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                _current = crc32(_current, hData.AddrOfPinnedObject().ToInt32()+offset, (uint)count);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            finally
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                hData.Free();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region Adler
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Implements a checksum generator that computes the Adler checksum on data
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public sealed class AdlerChecksum : ChecksumGeneratorBase
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region DLL imports
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern uint adler32(uint adler, int data, uint length);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the Adler checksum generator
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public AdlerChecksum() : base() {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the Adler checksum generator with a specified value
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="initialValue">The value to set the current checksum to</param>
 | 
				
			||||||
 | 
					        public AdlerChecksum(uint initialValue) : base(initialValue) {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with part of an array of bytes
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The data to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">Where in <c>data</c> to start updating</param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes from <c>data</c> to use</param>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentException">The sum of offset and count is larger than the length of <c>data</c></exception>
 | 
				
			||||||
 | 
					        /// <exception cref="NullReferenceException"><c>data</c> is a null reference</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentOutOfRangeException">Offset or count is negative.</exception>
 | 
				
			||||||
 | 
					        public override void Update(byte[] data, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (offset < 0 || count < 0) throw new ArgumentOutOfRangeException();
 | 
				
			||||||
 | 
					            if ((offset+count) > data.Length) throw new ArgumentException();
 | 
				
			||||||
 | 
					            GCHandle hData = GCHandle.Alloc(data, GCHandleType.Pinned);
 | 
				
			||||||
 | 
					            try
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                _current = adler32(_current, hData.AddrOfPinnedObject().ToInt32()+offset, (uint)count);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            finally
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                hData.Free();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,83 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.Diagnostics;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLib
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/// <summary>
 | 
				
			||||||
 | 
						/// This class implements a circular buffer
 | 
				
			||||||
 | 
						/// </summary>
 | 
				
			||||||
 | 
						internal class CircularBuffer
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
					        #region Private data
 | 
				
			||||||
 | 
					        private int _capacity;
 | 
				
			||||||
 | 
					        private int _head;
 | 
				
			||||||
 | 
					        private int _tail;
 | 
				
			||||||
 | 
					        private int _size;
 | 
				
			||||||
 | 
					        private byte[] _buffer;
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public CircularBuffer(int capacity)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            Debug.Assert( capacity > 0 );
 | 
				
			||||||
 | 
					            _buffer = new byte[capacity];
 | 
				
			||||||
 | 
					            _capacity = capacity;
 | 
				
			||||||
 | 
					            _head = 0;
 | 
				
			||||||
 | 
					            _tail = 0;
 | 
				
			||||||
 | 
					            _size = 0;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public int Size { get { return _size; } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public int Put(byte[] source, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            Debug.Assert( count > 0 );
 | 
				
			||||||
 | 
					            int trueCount = Math.Min(count, _capacity - Size);
 | 
				
			||||||
 | 
					            for (int i = 0; i < trueCount; ++i)
 | 
				
			||||||
 | 
					                _buffer[(_tail+i) % _capacity] = source[offset+i];
 | 
				
			||||||
 | 
					            _tail += trueCount;
 | 
				
			||||||
 | 
					            _tail %= _capacity;
 | 
				
			||||||
 | 
					            _size += trueCount;
 | 
				
			||||||
 | 
					            return trueCount;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public bool Put(byte b)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (Size == _capacity) // no room
 | 
				
			||||||
 | 
					                return false;
 | 
				
			||||||
 | 
					            _buffer[_tail++] = b;
 | 
				
			||||||
 | 
					            _tail %= _capacity;
 | 
				
			||||||
 | 
					            ++_size;
 | 
				
			||||||
 | 
					            return true;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public int Get(byte[] destination, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            int trueCount = Math.Min(count,Size);
 | 
				
			||||||
 | 
					            for (int i = 0; i < trueCount; ++i)
 | 
				
			||||||
 | 
					                destination[offset + i] = _buffer[(_head+i) % _capacity];
 | 
				
			||||||
 | 
					            _head += trueCount;
 | 
				
			||||||
 | 
					            _head %= _capacity;
 | 
				
			||||||
 | 
					            _size -= trueCount;
 | 
				
			||||||
 | 
					            return trueCount;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public int Get()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (Size == 0)
 | 
				
			||||||
 | 
					                return -1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            int result = (int)_buffer[_head++ % _capacity];
 | 
				
			||||||
 | 
					            --_size;
 | 
				
			||||||
 | 
					            return result;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,198 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.Runtime.InteropServices;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLib
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						/// <summary>
 | 
				
			||||||
 | 
						/// Implements the common functionality needed for all <see cref="Codec"/>s
 | 
				
			||||||
 | 
						/// </summary>
 | 
				
			||||||
 | 
						public abstract class CodecBase : Codec, IDisposable
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Data members
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Instance of the internal zlib buffer structure that is
 | 
				
			||||||
 | 
					        /// passed to all functions in the zlib dll
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        internal ZStream _ztream = new ZStream();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// True if the object instance has been disposed, false otherwise
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        protected bool _isDisposed = false;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// The size of the internal buffers
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        protected const int kBufferSize = 16384;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private byte[] _outBuffer = new byte[kBufferSize];
 | 
				
			||||||
 | 
					        private byte[] _inBuffer = new byte[kBufferSize];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private GCHandle _hInput;
 | 
				
			||||||
 | 
					        private GCHandle _hOutput;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private uint _checksum = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the <c>CodeBase</c> class.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
							public CodecBase()
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
					            try
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                _hInput = GCHandle.Alloc(_inBuffer, GCHandleType.Pinned);
 | 
				
			||||||
 | 
					                _hOutput = GCHandle.Alloc(_outBuffer, GCHandleType.Pinned);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            catch (Exception)
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                CleanUp(false);
 | 
				
			||||||
 | 
					                throw;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Codec Members
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Occurs when more processed data are available.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public event DataAvailableHandler DataAvailable;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Fires the <see cref="DataAvailable"/> event
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        protected void OnDataAvailable()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (_ztream.total_out > 0)
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                if (DataAvailable != null)
 | 
				
			||||||
 | 
					                    DataAvailable( _outBuffer, 0, (int)_ztream.total_out);
 | 
				
			||||||
 | 
					                resetOutput();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Adds more data to the codec to be processed.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">Byte array containing the data to be added to the codec</param>
 | 
				
			||||||
 | 
					        /// <remarks>Adding data may, or may not, raise the <c>DataAvailable</c> event</remarks>
 | 
				
			||||||
 | 
					        public void Add(byte[] data)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            Add(data,0,data.Length);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Adds more data to the codec to be processed.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">Byte array containing the data to be added to the codec</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">The index of the first byte to add from <c>data</c></param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes to add</param>
 | 
				
			||||||
 | 
					        /// <remarks>Adding data may, or may not, raise the <c>DataAvailable</c> event</remarks>
 | 
				
			||||||
 | 
					        /// <remarks>This must be implemented by a derived class</remarks>
 | 
				
			||||||
 | 
					        public abstract void Add(byte[] data, int offset, int count);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Finishes up any pending data that needs to be processed and handled.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <remarks>This must be implemented by a derived class</remarks>
 | 
				
			||||||
 | 
					        public abstract void Finish();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the checksum of the data that has been added so far
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public uint Checksum { get { return _checksum; } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Destructor & IDisposable stuff
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Destroys this instance
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        ~CodecBase()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            CleanUp(false);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Releases any unmanaged resources and calls the <see cref="CleanUp()"/> method of the derived class
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public void Dispose()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            CleanUp(true);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Performs any codec specific cleanup
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <remarks>This must be implemented by a derived class</remarks>
 | 
				
			||||||
 | 
					        protected abstract void CleanUp();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // performs the release of the handles and calls the derived CleanUp()
 | 
				
			||||||
 | 
					        private void CleanUp(bool isDisposing)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (!_isDisposed)
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                CleanUp();
 | 
				
			||||||
 | 
					                if (_hInput.IsAllocated)
 | 
				
			||||||
 | 
					                    _hInput.Free();
 | 
				
			||||||
 | 
					                if (_hOutput.IsAllocated)
 | 
				
			||||||
 | 
					                    _hOutput.Free();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                _isDisposed = true;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Helper methods
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Copies a number of bytes to the internal codec buffer - ready for processing
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The byte array that contains the data to copy</param>
 | 
				
			||||||
 | 
					        /// <param name="startIndex">The index of the first byte to copy</param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes to copy from <c>data</c></param>
 | 
				
			||||||
 | 
					        protected void copyInput(byte[] data, int startIndex, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            Array.Copy(data, startIndex, _inBuffer,0, count);
 | 
				
			||||||
 | 
					            _ztream.next_in = _hInput.AddrOfPinnedObject();
 | 
				
			||||||
 | 
					            _ztream.total_in = 0;
 | 
				
			||||||
 | 
					            _ztream.avail_in = (uint)count;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Resets the internal output buffers to a known state - ready for processing
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        protected void resetOutput()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            _ztream.total_out = 0;
 | 
				
			||||||
 | 
					            _ztream.avail_out = kBufferSize;
 | 
				
			||||||
 | 
					            _ztream.next_out = _hOutput.AddrOfPinnedObject();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the running checksum property
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="newSum">The new checksum value</param>
 | 
				
			||||||
 | 
					        protected void setChecksum(uint newSum)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            _checksum = newSum;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,106 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.Diagnostics;
 | 
				
			||||||
 | 
					using System.Runtime.InteropServices;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLib
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Implements a data compressor, using the deflate algorithm in the ZLib dll
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
						public sealed class Deflater : CodecBase
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
					        #region Dll imports
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl, CharSet=CharSet.Ansi)]
 | 
				
			||||||
 | 
					        private static extern int deflateInit_(ref ZStream sz, int level, string vs, int size);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int deflate(ref ZStream sz, int flush);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int deflateReset(ref ZStream sz);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int deflateEnd(ref ZStream sz);
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Constructs an new instance of the <c>Deflater</c>
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="level">The compression level to use for this <c>Deflater</c></param>
 | 
				
			||||||
 | 
							public Deflater(CompressLevel level) : base()
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
					            int retval = deflateInit_(ref _ztream, (int)level, Info.Version, Marshal.SizeOf(_ztream));
 | 
				
			||||||
 | 
					            if (retval != 0)
 | 
				
			||||||
 | 
					                throw new ZLibException(retval, "Could not initialize deflater");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            resetOutput();
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Adds more data to the codec to be processed.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">Byte array containing the data to be added to the codec</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">The index of the first byte to add from <c>data</c></param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes to add</param>
 | 
				
			||||||
 | 
					        /// <remarks>Adding data may, or may not, raise the <c>DataAvailable</c> event</remarks>
 | 
				
			||||||
 | 
					        public override void Add(byte[] data, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (data == null) throw new ArgumentNullException();
 | 
				
			||||||
 | 
					            if (offset < 0 || count < 0) throw new ArgumentOutOfRangeException();
 | 
				
			||||||
 | 
					            if ((offset+count) > data.Length) throw new ArgumentException();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            int total = count;
 | 
				
			||||||
 | 
					            int inputIndex = offset;
 | 
				
			||||||
 | 
					            int err = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            while (err >= 0 && inputIndex < total)
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                copyInput(data, inputIndex, Math.Min(total - inputIndex, kBufferSize));
 | 
				
			||||||
 | 
					                while (err >= 0 && _ztream.avail_in > 0)
 | 
				
			||||||
 | 
					                {
 | 
				
			||||||
 | 
					                    err = deflate(ref _ztream, (int)FlushTypes.None);
 | 
				
			||||||
 | 
					                    if (err == 0)
 | 
				
			||||||
 | 
					                        while (_ztream.avail_out == 0)
 | 
				
			||||||
 | 
					                        {
 | 
				
			||||||
 | 
					                            OnDataAvailable();
 | 
				
			||||||
 | 
					                            err = deflate(ref _ztream, (int)FlushTypes.None);
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                    inputIndex += (int)_ztream.total_in;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            setChecksum( _ztream.adler );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Finishes up any pending data that needs to be processed and handled.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public override void Finish()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            int err;
 | 
				
			||||||
 | 
					            do
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                err = deflate(ref _ztream, (int)FlushTypes.Finish);
 | 
				
			||||||
 | 
					                OnDataAvailable();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            while (err == 0);
 | 
				
			||||||
 | 
					            setChecksum( _ztream.adler );
 | 
				
			||||||
 | 
					            deflateReset(ref _ztream);
 | 
				
			||||||
 | 
					            resetOutput();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Closes the internal zlib deflate stream
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        protected override void CleanUp() { deflateEnd(ref _ztream); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,288 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.IO;
 | 
				
			||||||
 | 
					using System.Runtime.InteropServices;
 | 
				
			||||||
 | 
					using System.Text;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLib
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region Internal types
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Defines constants for the various flush types used with zlib
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    internal enum FlushTypes
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        None,  Partial,  Sync,  Full,  Finish,  Block
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region ZStream structure
 | 
				
			||||||
 | 
					    // internal mapping of the zlib zstream structure for marshalling
 | 
				
			||||||
 | 
					    [StructLayoutAttribute(LayoutKind.Sequential, Pack=4, Size=0, CharSet=CharSet.Ansi)]
 | 
				
			||||||
 | 
					    internal struct ZStream
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        public IntPtr next_in;
 | 
				
			||||||
 | 
					        public uint avail_in;
 | 
				
			||||||
 | 
					        public uint total_in;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public IntPtr next_out;
 | 
				
			||||||
 | 
					        public uint avail_out;
 | 
				
			||||||
 | 
					        public uint total_out;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [MarshalAs(UnmanagedType.LPStr)]
 | 
				
			||||||
 | 
					        string msg;
 | 
				
			||||||
 | 
					        uint state;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        uint zalloc;
 | 
				
			||||||
 | 
					        uint zfree;
 | 
				
			||||||
 | 
					        uint opaque;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        int data_type;
 | 
				
			||||||
 | 
					        public uint adler;
 | 
				
			||||||
 | 
					        uint reserved;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region Public enums
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Defines constants for the available compression levels in zlib
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public enum CompressLevel : int
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// The default compression level with a reasonable compromise between compression and speed
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        Default = -1,
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// No compression at all. The data are passed straight through.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        None = 0,
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// The maximum compression rate available.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        Best = 9,
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// The fastest available compression level.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        Fastest = 1
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region Exception classes
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// The exception that is thrown when an error occurs on the zlib dll
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public class ZLibException : ApplicationException
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the <see cref="ZLibException"/> class with a specified
 | 
				
			||||||
 | 
					        /// error message and error code
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="errorCode">The zlib error code that caused the exception</param>
 | 
				
			||||||
 | 
					        /// <param name="msg">A message that (hopefully) describes the error</param>
 | 
				
			||||||
 | 
					        public ZLibException(int errorCode, string msg) : base(String.Format("ZLib error {0} {1}", errorCode, msg))
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Initializes a new instance of the <see cref="ZLibException"/> class with a specified
 | 
				
			||||||
 | 
					        /// error code
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="errorCode">The zlib error code that caused the exception</param>
 | 
				
			||||||
 | 
					        public ZLibException(int errorCode) : base(String.Format("ZLib error {0}", errorCode))
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region Interfaces
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Declares methods and properties that enables a running checksum to be calculated
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public interface ChecksumGenerator
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the current value of the checksum
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        uint Value { get; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Clears the current checksum to 0
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        void Reset();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with an array of bytes
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The data to update the checksum with</param>
 | 
				
			||||||
 | 
					        void Update(byte[] data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with part of an array of bytes
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The data to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">Where in <c>data</c> to start updating</param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes from <c>data</c> to use</param>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentException">The sum of offset and count is larger than the length of <c>data</c></exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentNullException"><c>data</c> is a null reference</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentOutOfRangeException">Offset or count is negative.</exception>
 | 
				
			||||||
 | 
					        void Update(byte[] data, int offset, int count);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with the data from a string
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The string to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <remarks>The characters in the string are converted by the UTF-8 encoding</remarks>
 | 
				
			||||||
 | 
					        void Update(string data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Updates the current checksum with the data from a string, using a specific encoding
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">The string to update the checksum with</param>
 | 
				
			||||||
 | 
					        /// <param name="encoding">The encoding to use</param>
 | 
				
			||||||
 | 
					        void Update(string data, Encoding encoding);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Represents the method that will be called from a codec when new data
 | 
				
			||||||
 | 
					    /// are available.
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    /// <paramref name="data">The byte array containing the processed data</paramref>
 | 
				
			||||||
 | 
					    /// <paramref name="startIndex">The index of the first processed byte in <c>data</c></paramref>
 | 
				
			||||||
 | 
					    /// <paramref name="count">The number of processed bytes available</paramref>
 | 
				
			||||||
 | 
					    /// <remarks>On return from this method, the data may be overwritten, so grab it while you can.
 | 
				
			||||||
 | 
					    /// You cannot assume that startIndex will be zero.
 | 
				
			||||||
 | 
					    /// </remarks>
 | 
				
			||||||
 | 
					    public delegate void DataAvailableHandler(byte[] data, int startIndex, int count);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Declares methods and events for implementing compressors/decompressors
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public interface Codec
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Occurs when more processed data are available.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        event DataAvailableHandler DataAvailable;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Adds more data to the codec to be processed.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">Byte array containing the data to be added to the codec</param>
 | 
				
			||||||
 | 
					        /// <remarks>Adding data may, or may not, raise the <c>DataAvailable</c> event</remarks>
 | 
				
			||||||
 | 
					        void Add(byte[] data);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Adds more data to the codec to be processed.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">Byte array containing the data to be added to the codec</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">The index of the first byte to add from <c>data</c></param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes to add</param>
 | 
				
			||||||
 | 
					        /// <remarks>Adding data may, or may not, raise the <c>DataAvailable</c> event</remarks>
 | 
				
			||||||
 | 
					        void Add(byte[] data, int offset, int count);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Finishes up any pending data that needs to be processed and handled.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        void Finish();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the checksum of the data that has been added so far
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        uint Checksum { get; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #region Classes
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Encapsulates general information about the ZLib library
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public class Info
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region DLL imports
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern uint zlibCompileFlags();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern string zlibVersion();
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Private stuff
 | 
				
			||||||
 | 
					        private uint _flags;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // helper function that unpacks a bitsize mask
 | 
				
			||||||
 | 
					        private static int bitSize(uint bits)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            switch (bits)
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                case 0: return 16;
 | 
				
			||||||
 | 
					                case 1: return 32;
 | 
				
			||||||
 | 
					                case 2: return 64;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return -1;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Constructs an instance of the <c>Info</c> class.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public Info()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            _flags = zlibCompileFlags();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// True if the library is compiled with debug info
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public bool HasDebugInfo { get { return 0 != (_flags & 0x100); } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// True if the library is compiled with assembly optimizations
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public bool UsesAssemblyCode { get { return 0 != (_flags & 0x200); } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the size of the unsigned int that was compiled into Zlib
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public int SizeOfUInt { get { return bitSize(_flags & 3); } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the size of the unsigned long that was compiled into Zlib
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public int SizeOfULong { get { return bitSize((_flags >> 2) & 3); } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the size of the pointers that were compiled into Zlib
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public int SizeOfPointer { get { return bitSize((_flags >> 4) & 3); } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the size of the z_off_t type that was compiled into Zlib
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public int SizeOfOffset { get { return bitSize((_flags >> 6) & 3); } }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the version of ZLib as a string, e.g. "1.2.1"
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public static string Version { get { return zlibVersion(); } }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,141 @@
 | 
				
			||||||
 | 
					<VisualStudioProject>
 | 
				
			||||||
 | 
					    <CSHARP
 | 
				
			||||||
 | 
					        ProjectType = "Local"
 | 
				
			||||||
 | 
					        ProductVersion = "7.10.3077"
 | 
				
			||||||
 | 
					        SchemaVersion = "2.0"
 | 
				
			||||||
 | 
					        ProjectGuid = "{BB1EE0B1-1808-46CB-B786-949D91117FC5}"
 | 
				
			||||||
 | 
					    >
 | 
				
			||||||
 | 
					        <Build>
 | 
				
			||||||
 | 
					            <Settings
 | 
				
			||||||
 | 
					                ApplicationIcon = ""
 | 
				
			||||||
 | 
					                AssemblyKeyContainerName = ""
 | 
				
			||||||
 | 
					                AssemblyName = "DotZLib"
 | 
				
			||||||
 | 
					                AssemblyOriginatorKeyFile = ""
 | 
				
			||||||
 | 
					                DefaultClientScript = "JScript"
 | 
				
			||||||
 | 
					                DefaultHTMLPageLayout = "Grid"
 | 
				
			||||||
 | 
					                DefaultTargetSchema = "IE50"
 | 
				
			||||||
 | 
					                DelaySign = "false"
 | 
				
			||||||
 | 
					                OutputType = "Library"
 | 
				
			||||||
 | 
					                PreBuildEvent = ""
 | 
				
			||||||
 | 
					                PostBuildEvent = ""
 | 
				
			||||||
 | 
					                RootNamespace = "DotZLib"
 | 
				
			||||||
 | 
					                RunPostBuildEvent = "OnBuildSuccess"
 | 
				
			||||||
 | 
					                StartupObject = ""
 | 
				
			||||||
 | 
					            >
 | 
				
			||||||
 | 
					                <Config
 | 
				
			||||||
 | 
					                    Name = "Debug"
 | 
				
			||||||
 | 
					                    AllowUnsafeBlocks = "false"
 | 
				
			||||||
 | 
					                    BaseAddress = "285212672"
 | 
				
			||||||
 | 
					                    CheckForOverflowUnderflow = "false"
 | 
				
			||||||
 | 
					                    ConfigurationOverrideFile = ""
 | 
				
			||||||
 | 
					                    DefineConstants = "DEBUG;TRACE"
 | 
				
			||||||
 | 
					                    DocumentationFile = "docs\DotZLib.xml"
 | 
				
			||||||
 | 
					                    DebugSymbols = "true"
 | 
				
			||||||
 | 
					                    FileAlignment = "4096"
 | 
				
			||||||
 | 
					                    IncrementalBuild = "false"
 | 
				
			||||||
 | 
					                    NoStdLib = "false"
 | 
				
			||||||
 | 
					                    NoWarn = "1591"
 | 
				
			||||||
 | 
					                    Optimize = "false"
 | 
				
			||||||
 | 
					                    OutputPath = "bin\Debug\"
 | 
				
			||||||
 | 
					                    RegisterForComInterop = "false"
 | 
				
			||||||
 | 
					                    RemoveIntegerChecks = "false"
 | 
				
			||||||
 | 
					                    TreatWarningsAsErrors = "false"
 | 
				
			||||||
 | 
					                    WarningLevel = "4"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <Config
 | 
				
			||||||
 | 
					                    Name = "Release"
 | 
				
			||||||
 | 
					                    AllowUnsafeBlocks = "false"
 | 
				
			||||||
 | 
					                    BaseAddress = "285212672"
 | 
				
			||||||
 | 
					                    CheckForOverflowUnderflow = "false"
 | 
				
			||||||
 | 
					                    ConfigurationOverrideFile = ""
 | 
				
			||||||
 | 
					                    DefineConstants = "TRACE"
 | 
				
			||||||
 | 
					                    DocumentationFile = "docs\DotZLib.xml"
 | 
				
			||||||
 | 
					                    DebugSymbols = "false"
 | 
				
			||||||
 | 
					                    FileAlignment = "4096"
 | 
				
			||||||
 | 
					                    IncrementalBuild = "false"
 | 
				
			||||||
 | 
					                    NoStdLib = "false"
 | 
				
			||||||
 | 
					                    NoWarn = ""
 | 
				
			||||||
 | 
					                    Optimize = "true"
 | 
				
			||||||
 | 
					                    OutputPath = "bin\Release\"
 | 
				
			||||||
 | 
					                    RegisterForComInterop = "false"
 | 
				
			||||||
 | 
					                    RemoveIntegerChecks = "false"
 | 
				
			||||||
 | 
					                    TreatWarningsAsErrors = "false"
 | 
				
			||||||
 | 
					                    WarningLevel = "4"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					            </Settings>
 | 
				
			||||||
 | 
					            <References>
 | 
				
			||||||
 | 
					                <Reference
 | 
				
			||||||
 | 
					                    Name = "System"
 | 
				
			||||||
 | 
					                    AssemblyName = "System"
 | 
				
			||||||
 | 
					                    HintPath = "C:\WINNT\Microsoft.NET\Framework\v1.1.4322\System.dll"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <Reference
 | 
				
			||||||
 | 
					                    Name = "System.Data"
 | 
				
			||||||
 | 
					                    AssemblyName = "System.Data"
 | 
				
			||||||
 | 
					                    HintPath = "C:\WINNT\Microsoft.NET\Framework\v1.1.4322\System.Data.dll"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <Reference
 | 
				
			||||||
 | 
					                    Name = "System.XML"
 | 
				
			||||||
 | 
					                    AssemblyName = "System.Xml"
 | 
				
			||||||
 | 
					                    HintPath = "C:\WINNT\Microsoft.NET\Framework\v1.1.4322\System.XML.dll"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <Reference
 | 
				
			||||||
 | 
					                    Name = "nunit.framework"
 | 
				
			||||||
 | 
					                    AssemblyName = "nunit.framework"
 | 
				
			||||||
 | 
					                    HintPath = "E:\apps\NUnit V2.1\\bin\nunit.framework.dll"
 | 
				
			||||||
 | 
					                    AssemblyFolderKey = "hklm\dn\nunit.framework"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					            </References>
 | 
				
			||||||
 | 
					        </Build>
 | 
				
			||||||
 | 
					        <Files>
 | 
				
			||||||
 | 
					            <Include>
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "AssemblyInfo.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "ChecksumImpl.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "CircularBuffer.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "CodecBase.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "Deflater.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "DotZLib.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "GZipStream.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "Inflater.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					                <File
 | 
				
			||||||
 | 
					                    RelPath = "UnitTests.cs"
 | 
				
			||||||
 | 
					                    SubType = "Code"
 | 
				
			||||||
 | 
					                    BuildAction = "Compile"
 | 
				
			||||||
 | 
					                />
 | 
				
			||||||
 | 
					            </Include>
 | 
				
			||||||
 | 
					        </Files>
 | 
				
			||||||
 | 
					    </CSHARP>
 | 
				
			||||||
 | 
					</VisualStudioProject>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,301 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.IO;
 | 
				
			||||||
 | 
					using System.Runtime.InteropServices;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLib
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						/// <summary>
 | 
				
			||||||
 | 
						/// Implements a compressed <see cref="Stream"/>, in GZip (.gz) format.
 | 
				
			||||||
 | 
						/// </summary>
 | 
				
			||||||
 | 
						public class GZipStream : Stream, IDisposable
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
					        #region Dll Imports
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl, CharSet=CharSet.Ansi)]
 | 
				
			||||||
 | 
					        private static extern IntPtr gzopen(string name, string mode);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int gzclose(IntPtr gzFile);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int gzwrite(IntPtr gzFile, int data, int length);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int gzread(IntPtr gzFile, int data, int length);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int gzgetc(IntPtr gzFile);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int gzputc(IntPtr gzFile, int c);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Private data
 | 
				
			||||||
 | 
					        private IntPtr _gzFile;
 | 
				
			||||||
 | 
					        private bool _isDisposed = false;
 | 
				
			||||||
 | 
					        private bool _isWriting;
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Constructors
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Creates a new file as a writeable GZipStream
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="fileName">The name of the compressed file to create</param>
 | 
				
			||||||
 | 
					        /// <param name="level">The compression level to use when adding data</param>
 | 
				
			||||||
 | 
					        /// <exception cref="ZLibException">If an error occurred in the internal zlib function</exception>
 | 
				
			||||||
 | 
							public GZipStream(string fileName, CompressLevel level)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
					            _isWriting = true;
 | 
				
			||||||
 | 
					            _gzFile = gzopen(fileName, String.Format("wb{0}", (int)level));
 | 
				
			||||||
 | 
					            if (_gzFile == IntPtr.Zero)
 | 
				
			||||||
 | 
					                throw new ZLibException(-1, "Could not open " + fileName);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Opens an existing file as a readable GZipStream
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="fileName">The name of the file to open</param>
 | 
				
			||||||
 | 
					        /// <exception cref="ZLibException">If an error occurred in the internal zlib function</exception>
 | 
				
			||||||
 | 
					        public GZipStream(string fileName)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            _isWriting = false;
 | 
				
			||||||
 | 
					            _gzFile = gzopen(fileName, "rb");
 | 
				
			||||||
 | 
					            if (_gzFile == IntPtr.Zero)
 | 
				
			||||||
 | 
					                throw new ZLibException(-1, "Could not open " + fileName);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Access properties
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Returns true of this stream can be read from, false otherwise
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public override bool CanRead
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            get
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                return !_isWriting;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Returns false.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public override bool CanSeek
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            get
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                return false;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Returns true if this tsream is writeable, false otherwise
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public override bool CanWrite
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            get
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                return _isWriting;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Destructor & IDispose stuff
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Destroys this instance
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        ~GZipStream()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            cleanUp(false);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Closes the external file handle
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public void Dispose()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            cleanUp(true);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Does the actual closing of the file handle.
 | 
				
			||||||
 | 
					        private void cleanUp(bool isDisposing)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (!_isDisposed)
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                gzclose(_gzFile);
 | 
				
			||||||
 | 
					                _isDisposed = true;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Basic reading and writing
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Attempts to read a number of bytes from the stream.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="buffer">The destination data buffer</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">The index of the first destination byte in <c>buffer</c></param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes requested</param>
 | 
				
			||||||
 | 
					        /// <returns>The number of bytes read</returns>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentNullException">If <c>buffer</c> is null</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentOutOfRangeException">If <c>count</c> or <c>offset</c> are negative</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentException">If <c>offset</c>  + <c>count</c> is > buffer.Length</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="NotSupportedException">If this stream is not readable.</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ObjectDisposedException">If this stream has been disposed.</exception>
 | 
				
			||||||
 | 
					        public override int Read(byte[] buffer, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (!CanRead) throw new NotSupportedException();
 | 
				
			||||||
 | 
					            if (buffer == null) throw new ArgumentNullException();
 | 
				
			||||||
 | 
					            if (offset < 0 || count < 0) throw new ArgumentOutOfRangeException();
 | 
				
			||||||
 | 
					            if ((offset+count) > buffer.Length) throw new ArgumentException();
 | 
				
			||||||
 | 
					            if (_isDisposed) throw new ObjectDisposedException("GZipStream");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            GCHandle h = GCHandle.Alloc(buffer, GCHandleType.Pinned);
 | 
				
			||||||
 | 
					            int result;
 | 
				
			||||||
 | 
					            try
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                result = gzread(_gzFile, h.AddrOfPinnedObject().ToInt32() + offset, count);
 | 
				
			||||||
 | 
					                if (result < 0)
 | 
				
			||||||
 | 
					                    throw new IOException();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            finally
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                h.Free();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            return result;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Attempts to read a single byte from the stream.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <returns>The byte that was read, or -1 in case of error or End-Of-File</returns>
 | 
				
			||||||
 | 
					        public override int ReadByte()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (!CanRead) throw new NotSupportedException();
 | 
				
			||||||
 | 
					            if (_isDisposed) throw new ObjectDisposedException("GZipStream");
 | 
				
			||||||
 | 
					            return gzgetc(_gzFile);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Writes a number of bytes to the stream
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="buffer"></param>
 | 
				
			||||||
 | 
					        /// <param name="offset"></param>
 | 
				
			||||||
 | 
					        /// <param name="count"></param>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentNullException">If <c>buffer</c> is null</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentOutOfRangeException">If <c>count</c> or <c>offset</c> are negative</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ArgumentException">If <c>offset</c>  + <c>count</c> is > buffer.Length</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="NotSupportedException">If this stream is not writeable.</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ObjectDisposedException">If this stream has been disposed.</exception>
 | 
				
			||||||
 | 
					        public override void Write(byte[] buffer, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (!CanWrite) throw new NotSupportedException();
 | 
				
			||||||
 | 
					            if (buffer == null) throw new ArgumentNullException();
 | 
				
			||||||
 | 
					            if (offset < 0 || count < 0) throw new ArgumentOutOfRangeException();
 | 
				
			||||||
 | 
					            if ((offset+count) > buffer.Length) throw new ArgumentException();
 | 
				
			||||||
 | 
					            if (_isDisposed) throw new ObjectDisposedException("GZipStream");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            GCHandle h = GCHandle.Alloc(buffer, GCHandleType.Pinned);
 | 
				
			||||||
 | 
					            try
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                int result = gzwrite(_gzFile, h.AddrOfPinnedObject().ToInt32() + offset, count);
 | 
				
			||||||
 | 
					                if (result < 0)
 | 
				
			||||||
 | 
					                    throw new IOException();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            finally
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                h.Free();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Writes a single byte to the stream
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="value">The byte to add to the stream.</param>
 | 
				
			||||||
 | 
					        /// <exception cref="NotSupportedException">If this stream is not writeable.</exception>
 | 
				
			||||||
 | 
					        /// <exception cref="ObjectDisposedException">If this stream has been disposed.</exception>
 | 
				
			||||||
 | 
					        public override void WriteByte(byte value)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (!CanWrite) throw new NotSupportedException();
 | 
				
			||||||
 | 
					            if (_isDisposed) throw new ObjectDisposedException("GZipStream");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            int result = gzputc(_gzFile, (int)value);
 | 
				
			||||||
 | 
					            if (result < 0)
 | 
				
			||||||
 | 
					                throw new IOException();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Position & length stuff
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Not supported.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="value"></param>
 | 
				
			||||||
 | 
					        /// <exception cref="NotSupportedException">Always thrown</exception>
 | 
				
			||||||
 | 
					        public override void SetLength(long value)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            throw new NotSupportedException();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        ///  Not supported.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="offset"></param>
 | 
				
			||||||
 | 
					        /// <param name="origin"></param>
 | 
				
			||||||
 | 
					        /// <returns></returns>
 | 
				
			||||||
 | 
					        /// <exception cref="NotSupportedException">Always thrown</exception>
 | 
				
			||||||
 | 
					        public override long Seek(long offset, SeekOrigin origin)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            throw new NotSupportedException();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Flushes the <c>GZipStream</c>.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <remarks>In this implementation, this method does nothing. This is because excessive
 | 
				
			||||||
 | 
					        /// flushing may degrade the achievable compression rates.</remarks>
 | 
				
			||||||
 | 
					        public override void Flush()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            // left empty on purpose
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets/sets the current position in the <c>GZipStream</c>. Not supported.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <remarks>In this implementation this property is not supported</remarks>
 | 
				
			||||||
 | 
					        /// <exception cref="NotSupportedException">Always thrown</exception>
 | 
				
			||||||
 | 
					        public override long Position
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            get
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                throw new NotSupportedException();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            set
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                throw new NotSupportedException();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Gets the size of the stream. Not supported.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <remarks>In this implementation this property is not supported</remarks>
 | 
				
			||||||
 | 
					        /// <exception cref="NotSupportedException">Always thrown</exception>
 | 
				
			||||||
 | 
					        public override long Length
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            get
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                throw new NotSupportedException();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,105 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.Diagnostics;
 | 
				
			||||||
 | 
					using System.Runtime.InteropServices;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLib
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /// <summary>
 | 
				
			||||||
 | 
					    /// Implements a data decompressor, using the inflate algorithm in the ZLib dll
 | 
				
			||||||
 | 
					    /// </summary>
 | 
				
			||||||
 | 
					    public class Inflater : CodecBase
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
					        #region Dll imports
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl, CharSet=CharSet.Ansi)]
 | 
				
			||||||
 | 
					        private static extern int inflateInit_(ref ZStream sz, string vs, int size);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int inflate(ref ZStream sz, int flush);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int inflateReset(ref ZStream sz);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [DllImport("ZLIB1.dll", CallingConvention=CallingConvention.Cdecl)]
 | 
				
			||||||
 | 
					        private static extern int inflateEnd(ref ZStream sz);
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Constructs an new instance of the <c>Inflater</c>
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public Inflater() : base()
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
					            int retval = inflateInit_(ref _ztream, Info.Version, Marshal.SizeOf(_ztream));
 | 
				
			||||||
 | 
					            if (retval != 0)
 | 
				
			||||||
 | 
					                throw new ZLibException(retval, "Could not initialize inflater");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            resetOutput();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Adds more data to the codec to be processed.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        /// <param name="data">Byte array containing the data to be added to the codec</param>
 | 
				
			||||||
 | 
					        /// <param name="offset">The index of the first byte to add from <c>data</c></param>
 | 
				
			||||||
 | 
					        /// <param name="count">The number of bytes to add</param>
 | 
				
			||||||
 | 
					        /// <remarks>Adding data may, or may not, raise the <c>DataAvailable</c> event</remarks>
 | 
				
			||||||
 | 
					        public override void Add(byte[] data, int offset, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (data == null) throw new ArgumentNullException();
 | 
				
			||||||
 | 
					            if (offset < 0 || count < 0) throw new ArgumentOutOfRangeException();
 | 
				
			||||||
 | 
					            if ((offset+count) > data.Length) throw new ArgumentException();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            int total = count;
 | 
				
			||||||
 | 
					            int inputIndex = offset;
 | 
				
			||||||
 | 
					            int err = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            while (err >= 0 && inputIndex < total)
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                copyInput(data, inputIndex, Math.Min(total - inputIndex, kBufferSize));
 | 
				
			||||||
 | 
					                err = inflate(ref _ztream, (int)FlushTypes.None);
 | 
				
			||||||
 | 
					                if (err == 0)
 | 
				
			||||||
 | 
					                    while (_ztream.avail_out == 0)
 | 
				
			||||||
 | 
					                    {
 | 
				
			||||||
 | 
					                        OnDataAvailable();
 | 
				
			||||||
 | 
					                        err = inflate(ref _ztream, (int)FlushTypes.None);
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					                inputIndex += (int)_ztream.total_in;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            setChecksum( _ztream.adler );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Finishes up any pending data that needs to be processed and handled.
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        public override void Finish()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            int err;
 | 
				
			||||||
 | 
					            do
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                err = inflate(ref _ztream, (int)FlushTypes.Finish);
 | 
				
			||||||
 | 
					                OnDataAvailable();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            while (err == 0);
 | 
				
			||||||
 | 
					            setChecksum( _ztream.adler );
 | 
				
			||||||
 | 
					            inflateReset(ref _ztream);
 | 
				
			||||||
 | 
					            resetOutput();
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        /// <summary>
 | 
				
			||||||
 | 
					        /// Closes the internal zlib inflate stream
 | 
				
			||||||
 | 
					        /// </summary>
 | 
				
			||||||
 | 
					        protected override void CleanUp() { inflateEnd(ref _ztream); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,274 @@
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// © Copyright Henrik Ravn 2004
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					// (See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					using System;
 | 
				
			||||||
 | 
					using System.Collections;
 | 
				
			||||||
 | 
					using System.IO;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// uncomment the define below to include unit tests
 | 
				
			||||||
 | 
					//#define nunit
 | 
				
			||||||
 | 
					#if nunit
 | 
				
			||||||
 | 
					using NUnit.Framework;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Unit tests for the DotZLib class library
 | 
				
			||||||
 | 
					// ----------------------------------------
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					// Use this with NUnit 2 from http://www.nunit.org
 | 
				
			||||||
 | 
					//
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					namespace DotZLibTests
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    using DotZLib;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // helper methods
 | 
				
			||||||
 | 
					    internal class Utils
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        public static bool byteArrEqual( byte[] lhs, byte[] rhs )
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            if (lhs.Length != rhs.Length)
 | 
				
			||||||
 | 
					                return false;
 | 
				
			||||||
 | 
					            for (int i = lhs.Length-1; i >= 0; --i)
 | 
				
			||||||
 | 
					                if (lhs[i] != rhs[i])
 | 
				
			||||||
 | 
					                    return false;
 | 
				
			||||||
 | 
					            return true;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    [TestFixture]
 | 
				
			||||||
 | 
					    public class CircBufferTests
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region Circular buffer tests
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void SinglePutGet()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            CircularBuffer buf = new CircularBuffer(10);
 | 
				
			||||||
 | 
					            Assert.AreEqual( 0, buf.Size );
 | 
				
			||||||
 | 
					            Assert.AreEqual( -1, buf.Get() );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            Assert.IsTrue(buf.Put( 1 ));
 | 
				
			||||||
 | 
					            Assert.AreEqual( 1, buf.Size );
 | 
				
			||||||
 | 
					            Assert.AreEqual( 1, buf.Get() );
 | 
				
			||||||
 | 
					            Assert.AreEqual( 0, buf.Size );
 | 
				
			||||||
 | 
					            Assert.AreEqual( -1, buf.Get() );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void BlockPutGet()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            CircularBuffer buf = new CircularBuffer(10);
 | 
				
			||||||
 | 
					            byte[] arr = {1,2,3,4,5,6,7,8,9,10};
 | 
				
			||||||
 | 
					            Assert.AreEqual( 10, buf.Put(arr,0,10) );
 | 
				
			||||||
 | 
					            Assert.AreEqual( 10, buf.Size );
 | 
				
			||||||
 | 
					            Assert.IsFalse( buf.Put(11) );
 | 
				
			||||||
 | 
					            Assert.AreEqual( 1, buf.Get() );
 | 
				
			||||||
 | 
					            Assert.IsTrue( buf.Put(11) );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            byte[] arr2 = (byte[])arr.Clone();
 | 
				
			||||||
 | 
					            Assert.AreEqual( 9, buf.Get(arr2,1,9) );
 | 
				
			||||||
 | 
					            Assert.IsTrue( Utils.byteArrEqual(arr,arr2) );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    [TestFixture]
 | 
				
			||||||
 | 
					    public class ChecksumTests
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region CRC32 Tests
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void CRC32_Null()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            CRC32Checksum crc32 = new CRC32Checksum();
 | 
				
			||||||
 | 
					            Assert.AreEqual( 0, crc32.Value );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            crc32 = new CRC32Checksum(1);
 | 
				
			||||||
 | 
					            Assert.AreEqual( 1, crc32.Value );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            crc32 = new CRC32Checksum(556);
 | 
				
			||||||
 | 
					            Assert.AreEqual( 556, crc32.Value );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void CRC32_Data()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            CRC32Checksum crc32 = new CRC32Checksum();
 | 
				
			||||||
 | 
					            byte[] data = { 1,2,3,4,5,6,7 };
 | 
				
			||||||
 | 
					            crc32.Update(data);
 | 
				
			||||||
 | 
					            Assert.AreEqual( 0x70e46888, crc32.Value  );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            crc32 = new CRC32Checksum();
 | 
				
			||||||
 | 
					            crc32.Update("penguin");
 | 
				
			||||||
 | 
					            Assert.AreEqual( 0x0e5c1a120, crc32.Value );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            crc32 = new CRC32Checksum(1);
 | 
				
			||||||
 | 
					            crc32.Update("penguin");
 | 
				
			||||||
 | 
					            Assert.AreEqual(0x43b6aa94, crc32.Value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Adler tests
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void Adler_Null()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            AdlerChecksum adler = new AdlerChecksum();
 | 
				
			||||||
 | 
					            Assert.AreEqual(0, adler.Value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            adler = new AdlerChecksum(1);
 | 
				
			||||||
 | 
					            Assert.AreEqual( 1, adler.Value );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            adler = new AdlerChecksum(556);
 | 
				
			||||||
 | 
					            Assert.AreEqual( 556, adler.Value );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void Adler_Data()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            AdlerChecksum adler = new AdlerChecksum(1);
 | 
				
			||||||
 | 
					            byte[] data = { 1,2,3,4,5,6,7 };
 | 
				
			||||||
 | 
					            adler.Update(data);
 | 
				
			||||||
 | 
					            Assert.AreEqual( 0x5b001d, adler.Value  );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            adler = new AdlerChecksum();
 | 
				
			||||||
 | 
					            adler.Update("penguin");
 | 
				
			||||||
 | 
					            Assert.AreEqual(0x0bcf02f6, adler.Value );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            adler = new AdlerChecksum(1);
 | 
				
			||||||
 | 
					            adler.Update("penguin");
 | 
				
			||||||
 | 
					            Assert.AreEqual(0x0bd602f7, adler.Value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    [TestFixture]
 | 
				
			||||||
 | 
					    public class InfoTests
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region Info tests
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void Info_Version()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            Info info = new Info();
 | 
				
			||||||
 | 
					            Assert.AreEqual("1.3.1", Info.Version);
 | 
				
			||||||
 | 
					            Assert.AreEqual(32, info.SizeOfUInt);
 | 
				
			||||||
 | 
					            Assert.AreEqual(32, info.SizeOfULong);
 | 
				
			||||||
 | 
					            Assert.AreEqual(32, info.SizeOfPointer);
 | 
				
			||||||
 | 
					            Assert.AreEqual(32, info.SizeOfOffset);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    [TestFixture]
 | 
				
			||||||
 | 
					    public class DeflateInflateTests
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region Deflate tests
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void Deflate_Init()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            using (Deflater def = new Deflater(CompressLevel.Default))
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private ArrayList compressedData = new ArrayList();
 | 
				
			||||||
 | 
					        private uint adler1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private ArrayList uncompressedData = new ArrayList();
 | 
				
			||||||
 | 
					        private uint adler2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        public void CDataAvail(byte[] data, int startIndex, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            for (int i = 0; i < count; ++i)
 | 
				
			||||||
 | 
					                compressedData.Add(data[i+startIndex]);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void Deflate_Compress()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            compressedData.Clear();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            byte[] testData = new byte[35000];
 | 
				
			||||||
 | 
					            for (int i = 0; i < testData.Length; ++i)
 | 
				
			||||||
 | 
					                testData[i] = 5;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            using (Deflater def = new Deflater((CompressLevel)5))
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                def.DataAvailable += new DataAvailableHandler(CDataAvail);
 | 
				
			||||||
 | 
					                def.Add(testData);
 | 
				
			||||||
 | 
					                def.Finish();
 | 
				
			||||||
 | 
					                adler1 = def.Checksum;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #region Inflate tests
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void Inflate_Init()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            using (Inflater inf = new Inflater())
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        private void DDataAvail(byte[] data, int startIndex, int count)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            for (int i = 0; i < count; ++i)
 | 
				
			||||||
 | 
					                uncompressedData.Add(data[i+startIndex]);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void Inflate_Expand()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            uncompressedData.Clear();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            using (Inflater inf = new Inflater())
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                inf.DataAvailable += new DataAvailableHandler(DDataAvail);
 | 
				
			||||||
 | 
					                inf.Add((byte[])compressedData.ToArray(typeof(byte)));
 | 
				
			||||||
 | 
					                inf.Finish();
 | 
				
			||||||
 | 
					                adler2 = inf.Checksum;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            Assert.AreEqual( adler1, adler2 );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    [TestFixture]
 | 
				
			||||||
 | 
					    public class GZipStreamTests
 | 
				
			||||||
 | 
					    {
 | 
				
			||||||
 | 
					        #region GZipStream test
 | 
				
			||||||
 | 
					        [Test]
 | 
				
			||||||
 | 
					        public void GZipStream_WriteRead()
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            using (GZipStream gzOut = new GZipStream("gzstream.gz", CompressLevel.Best))
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                BinaryWriter writer = new BinaryWriter(gzOut);
 | 
				
			||||||
 | 
					                writer.Write("hi there");
 | 
				
			||||||
 | 
					                writer.Write(Math.PI);
 | 
				
			||||||
 | 
					                writer.Write(42);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            using (GZipStream gzIn = new GZipStream("gzstream.gz"))
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                BinaryReader reader = new BinaryReader(gzIn);
 | 
				
			||||||
 | 
					                string s = reader.ReadString();
 | 
				
			||||||
 | 
					                Assert.AreEqual("hi there",s);
 | 
				
			||||||
 | 
					                double d = reader.ReadDouble();
 | 
				
			||||||
 | 
					                Assert.AreEqual(Math.PI, d);
 | 
				
			||||||
 | 
					                int i = reader.ReadInt32();
 | 
				
			||||||
 | 
					                Assert.AreEqual(42,i);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        #endregion
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,23 @@
 | 
				
			||||||
 | 
					Boost Software License - Version 1.0 - August 17th, 2003
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Permission is hereby granted, free of charge, to any person or organization
 | 
				
			||||||
 | 
					obtaining a copy of the software and accompanying documentation covered by
 | 
				
			||||||
 | 
					this license (the "Software") to use, reproduce, display, distribute,
 | 
				
			||||||
 | 
					execute, and transmit the Software, and to prepare derivative works of the
 | 
				
			||||||
 | 
					Software, and to permit third-parties to whom the Software is furnished to
 | 
				
			||||||
 | 
					do so, all subject to the following:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The copyright notices in the Software and this entire statement, including
 | 
				
			||||||
 | 
					the above license grant, this restriction and the following disclaimer,
 | 
				
			||||||
 | 
					must be included in all copies of the Software, in whole or in part, and
 | 
				
			||||||
 | 
					all derivative works of the Software, unless such copies or derivative
 | 
				
			||||||
 | 
					works are solely in the form of machine-executable object code generated by
 | 
				
			||||||
 | 
					a source language processor.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
				
			||||||
 | 
					IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
				
			||||||
 | 
					FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
 | 
				
			||||||
 | 
					SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
 | 
				
			||||||
 | 
					FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
 | 
				
			||||||
 | 
					ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 | 
				
			||||||
 | 
					DEALINGS IN THE SOFTWARE.
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,58 @@
 | 
				
			||||||
 | 
					This directory contains a .Net wrapper class library for the ZLib1.dll
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					The wrapper includes support for inflating/deflating memory buffers,
 | 
				
			||||||
 | 
					.Net streaming wrappers for the gz streams part of zlib, and wrappers
 | 
				
			||||||
 | 
					for the checksum parts of zlib. See DotZLib/UnitTests.cs for examples.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Directory structure:
 | 
				
			||||||
 | 
					--------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LICENSE_1_0.txt       - License file.
 | 
				
			||||||
 | 
					readme.txt            - This file.
 | 
				
			||||||
 | 
					DotZLib.chm           - Class library documentation
 | 
				
			||||||
 | 
					DotZLib.build         - NAnt build file
 | 
				
			||||||
 | 
					DotZLib.sln           - Microsoft Visual Studio 2003 solution file
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					DotZLib\*.cs          - Source files for the class library
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Unit tests:
 | 
				
			||||||
 | 
					-----------
 | 
				
			||||||
 | 
					The file DotZLib/UnitTests.cs contains unit tests for use with NUnit 2.1 or higher.
 | 
				
			||||||
 | 
					To include unit tests in the build, define nunit before building.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Build instructions:
 | 
				
			||||||
 | 
					-------------------
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					1. Using Visual Studio.Net 2003:
 | 
				
			||||||
 | 
					   Open DotZLib.sln in VS.Net and build from there. Output file (DotZLib.dll)
 | 
				
			||||||
 | 
					   will be found ./DotZLib/bin/release or ./DotZLib/bin/debug, depending on
 | 
				
			||||||
 | 
					   you are building the release or debug version of the library. Check
 | 
				
			||||||
 | 
					   DotZLib/UnitTests.cs for instructions on how to include unit tests in the
 | 
				
			||||||
 | 
					   build.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					2. Using NAnt:
 | 
				
			||||||
 | 
					   Open a command prompt with access to the build environment and run nant
 | 
				
			||||||
 | 
					   in the same directory as the DotZLib.build file.
 | 
				
			||||||
 | 
					   You can define 2 properties on the nant command-line to control the build:
 | 
				
			||||||
 | 
					   debug={true|false} to toggle between release/debug builds (default=true).
 | 
				
			||||||
 | 
					   nunit={true|false} to include or exclude unit tests (default=true).
 | 
				
			||||||
 | 
					   Also the target clean will remove binaries.
 | 
				
			||||||
 | 
					   Output file (DotZLib.dll) will be found in either ./DotZLib/bin/release
 | 
				
			||||||
 | 
					   or ./DotZLib/bin/debug, depending on whether you are building the release
 | 
				
			||||||
 | 
					   or debug version of the library.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   Examples:
 | 
				
			||||||
 | 
					     nant -D:debug=false -D:nunit=false
 | 
				
			||||||
 | 
					       will build a release mode version of the library without unit tests.
 | 
				
			||||||
 | 
					     nant
 | 
				
			||||||
 | 
					       will build a debug version of the library with unit tests
 | 
				
			||||||
 | 
					     nant clean
 | 
				
			||||||
 | 
					       will remove all previously built files.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					---------------------------------
 | 
				
			||||||
 | 
					Copyright (c) Henrik Ravn 2004
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Use, modification and distribution are subject to the Boost Software License, Version 1.0.
 | 
				
			||||||
 | 
					(See accompanying file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,574 @@
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;uInt longest_match_x64(
 | 
				
			||||||
 | 
					;    deflate_state *s,
 | 
				
			||||||
 | 
					;    IPos cur_match);                             // current match 
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					; gvmat64.S -- Asm portion of the optimized longest_match for 32 bits x86_64
 | 
				
			||||||
 | 
					;  (AMD64 on Athlon 64, Opteron, Phenom
 | 
				
			||||||
 | 
					;     and Intel EM64T on Pentium 4 with EM64T, Pentium D, Core 2 Duo, Core I5/I7)
 | 
				
			||||||
 | 
					; this file is translation from gvmat64.asm to GCC 4.x (for Linux, Mac XCode)
 | 
				
			||||||
 | 
					; Copyright (C) 1995-2010 Jean-loup Gailly, Brian Raiter and Gilles Vollant.
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					; File written by Gilles Vollant, by converting to assembly the longest_match
 | 
				
			||||||
 | 
					;  from Jean-loup Gailly in deflate.c of zLib and infoZip zip.
 | 
				
			||||||
 | 
					;  and by taking inspiration on asm686 with masm, optimised assembly code
 | 
				
			||||||
 | 
					;        from Brian Raiter, written 1998
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					;  This software is provided 'as-is', without any express or implied
 | 
				
			||||||
 | 
					;  warranty.  In no event will the authors be held liable for any damages
 | 
				
			||||||
 | 
					;  arising from the use of this software.
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					;  Permission is granted to anyone to use this software for any purpose,
 | 
				
			||||||
 | 
					;  including commercial applications, and to alter it and redistribute it
 | 
				
			||||||
 | 
					;  freely, subject to the following restrictions:
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					;  1. The origin of this software must not be misrepresented; you must not
 | 
				
			||||||
 | 
					;     claim that you wrote the original software. If you use this software
 | 
				
			||||||
 | 
					;     in a product, an acknowledgment in the product documentation would be
 | 
				
			||||||
 | 
					;     appreciated but is not required.
 | 
				
			||||||
 | 
					;  2. Altered source versions must be plainly marked as such, and must not be
 | 
				
			||||||
 | 
					;     misrepresented as being the original software
 | 
				
			||||||
 | 
					;  3. This notice may not be removed or altered from any source distribution.
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					;         http://www.zlib.net
 | 
				
			||||||
 | 
					;         http://www.winimage.com/zLibDll
 | 
				
			||||||
 | 
					;         http://www.muppetlabs.com/~breadbox/software/assembly.html
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					; to compile this file for zLib, I use option:
 | 
				
			||||||
 | 
					;   gcc -c -arch x86_64 gvmat64.S
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					;uInt longest_match(s, cur_match)
 | 
				
			||||||
 | 
					;    deflate_state *s;
 | 
				
			||||||
 | 
					;    IPos cur_match;                             // current match /
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					; with XCode for Mac, I had strange error with some jump on intel syntax
 | 
				
			||||||
 | 
					; this is why BEFORE_JMP and AFTER_JMP are used
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define BEFORE_JMP .att_syntax
 | 
				
			||||||
 | 
					#define AFTER_JMP .intel_syntax noprefix
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef NO_UNDERLINE
 | 
				
			||||||
 | 
					#	define	match_init	_match_init
 | 
				
			||||||
 | 
					#	define	longest_match	_longest_match
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.intel_syntax noprefix
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.globl	match_init, longest_match
 | 
				
			||||||
 | 
					.text
 | 
				
			||||||
 | 
					longest_match:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define LocalVarsSize 96
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					; register used : rax,rbx,rcx,rdx,rsi,rdi,r8,r9,r10,r11,r12
 | 
				
			||||||
 | 
					; free register :  r14,r15
 | 
				
			||||||
 | 
					; register can be saved : rsp
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define chainlenwmask     (rsp + 8 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define nicematch         (rsp + 16 - LocalVarsSize)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define save_rdi        (rsp + 24 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define save_rsi        (rsp + 32 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define save_rbx        (rsp + 40 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define save_rbp        (rsp + 48 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define save_r12        (rsp + 56 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define save_r13        (rsp + 64 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define save_r14        (rsp + 72 - LocalVarsSize)
 | 
				
			||||||
 | 
					#define save_r15        (rsp + 80 - LocalVarsSize)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;  all the +4 offsets are due to the addition of pending_buf_size (in zlib
 | 
				
			||||||
 | 
					;  in the deflate_state structure since the asm code was first written
 | 
				
			||||||
 | 
					;  (if you compile with zlib 1.0.4 or older, remove the +4).
 | 
				
			||||||
 | 
					;  Note : these value are good with a 8 bytes boundary pack structure
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define    MAX_MATCH              258
 | 
				
			||||||
 | 
					#define    MIN_MATCH              3
 | 
				
			||||||
 | 
					#define    MIN_LOOKAHEAD          (MAX_MATCH+MIN_MATCH+1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;;; Offsets for fields in the deflate_state structure. These numbers
 | 
				
			||||||
 | 
					;;; are calculated from the definition of deflate_state, with the
 | 
				
			||||||
 | 
					;;; assumption that the compiler will dword-align the fields. (Thus,
 | 
				
			||||||
 | 
					;;; changing the definition of deflate_state could easily cause this
 | 
				
			||||||
 | 
					;;; program to crash horribly, without so much as a warning at
 | 
				
			||||||
 | 
					;;; compile time. Sigh.)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					;  all the +zlib1222add offsets are due to the addition of fields
 | 
				
			||||||
 | 
					;  in zlib in the deflate_state structure since the asm code was first written
 | 
				
			||||||
 | 
					;  (if you compile with zlib 1.0.4 or older, use "zlib1222add equ (-4)").
 | 
				
			||||||
 | 
					;  (if you compile with zlib between 1.0.5 and 1.2.2.1, use "zlib1222add equ 0").
 | 
				
			||||||
 | 
					;  if you compile with zlib 1.2.2.2 or later , use "zlib1222add equ 8").
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* you can check the structure offset by running
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <stdlib.h>
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					#include "deflate.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void print_depl()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					deflate_state ds;
 | 
				
			||||||
 | 
					deflate_state *s=&ds;
 | 
				
			||||||
 | 
					printf("size pointer=%u\n",(int)sizeof(void*));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					printf("#define dsWSize         %u\n",(int)(((char*)&(s->w_size))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsWMask         %u\n",(int)(((char*)&(s->w_mask))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsWindow        %u\n",(int)(((char*)&(s->window))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsPrev          %u\n",(int)(((char*)&(s->prev))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsMatchLen      %u\n",(int)(((char*)&(s->match_length))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsPrevMatch     %u\n",(int)(((char*)&(s->prev_match))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsStrStart      %u\n",(int)(((char*)&(s->strstart))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsMatchStart    %u\n",(int)(((char*)&(s->match_start))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsLookahead     %u\n",(int)(((char*)&(s->lookahead))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsPrevLen       %u\n",(int)(((char*)&(s->prev_length))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsMaxChainLen   %u\n",(int)(((char*)&(s->max_chain_length))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsGoodMatch     %u\n",(int)(((char*)&(s->good_match))-((char*)s)));
 | 
				
			||||||
 | 
					printf("#define dsNiceMatch     %u\n",(int)(((char*)&(s->nice_match))-((char*)s)));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define dsWSize          68
 | 
				
			||||||
 | 
					#define dsWMask          76
 | 
				
			||||||
 | 
					#define dsWindow         80
 | 
				
			||||||
 | 
					#define dsPrev           96
 | 
				
			||||||
 | 
					#define dsMatchLen       144
 | 
				
			||||||
 | 
					#define dsPrevMatch      148
 | 
				
			||||||
 | 
					#define dsStrStart       156
 | 
				
			||||||
 | 
					#define dsMatchStart     160
 | 
				
			||||||
 | 
					#define dsLookahead      164
 | 
				
			||||||
 | 
					#define dsPrevLen        168
 | 
				
			||||||
 | 
					#define dsMaxChainLen    172
 | 
				
			||||||
 | 
					#define dsGoodMatch      188
 | 
				
			||||||
 | 
					#define dsNiceMatch      192
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define window_size      [ rcx + dsWSize]
 | 
				
			||||||
 | 
					#define WMask            [ rcx + dsWMask]
 | 
				
			||||||
 | 
					#define window_ad        [ rcx + dsWindow]
 | 
				
			||||||
 | 
					#define prev_ad          [ rcx + dsPrev]
 | 
				
			||||||
 | 
					#define strstart         [ rcx + dsStrStart]
 | 
				
			||||||
 | 
					#define match_start      [ rcx + dsMatchStart]
 | 
				
			||||||
 | 
					#define Lookahead        [ rcx + dsLookahead] //; 0ffffffffh on infozip
 | 
				
			||||||
 | 
					#define prev_length      [ rcx + dsPrevLen]
 | 
				
			||||||
 | 
					#define max_chain_length [ rcx + dsMaxChainLen]
 | 
				
			||||||
 | 
					#define good_match       [ rcx + dsGoodMatch]
 | 
				
			||||||
 | 
					#define nice_match       [ rcx + dsNiceMatch]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					; windows:
 | 
				
			||||||
 | 
					; parameter 1 in rcx(deflate state s), param 2 in rdx (cur match)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					; see http://weblogs.asp.net/oldnewthing/archive/2004/01/14/58579.aspx and
 | 
				
			||||||
 | 
					; http://msdn.microsoft.com/library/en-us/kmarch/hh/kmarch/64bitAMD_8e951dd2-ee77-4728-8702-55ce4b5dd24a.xml.asp
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					; All registers must be preserved across the call, except for
 | 
				
			||||||
 | 
					;   rax, rcx, rdx, r8, r9, r10, and r11, which are scratch.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					;
 | 
				
			||||||
 | 
					; gcc on macosx-linux:
 | 
				
			||||||
 | 
					; see http://www.x86-64.org/documentation/abi-0.99.pdf
 | 
				
			||||||
 | 
					; param 1 in rdi, param 2 in rsi
 | 
				
			||||||
 | 
					; rbx, rsp, rbp, r12 to r15 must be preserved
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					;;; Save registers that the compiler may be using, and adjust esp to
 | 
				
			||||||
 | 
					;;; make room for our stack frame.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					;;; Retrieve the function arguments. r8d will hold cur_match
 | 
				
			||||||
 | 
					;;; throughout the entire function. edx will hold the pointer to the
 | 
				
			||||||
 | 
					;;; deflate_state structure during the function's setup (before
 | 
				
			||||||
 | 
					;;; entering the main loop.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					; ms: parameter 1 in rcx (deflate_state* s), param 2 in edx -> r8 (cur match)
 | 
				
			||||||
 | 
					; mac: param 1 in rdi, param 2 rsi
 | 
				
			||||||
 | 
					; this clear high 32 bits of r8, which can be garbage in both r8 and rdx
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					        mov [save_rbx],rbx
 | 
				
			||||||
 | 
					        mov [save_rbp],rbp
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov rcx,rdi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov r8d,esi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov [save_r12],r12
 | 
				
			||||||
 | 
					        mov [save_r13],r13
 | 
				
			||||||
 | 
					        mov [save_r14],r14
 | 
				
			||||||
 | 
					        mov [save_r15],r15
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; uInt wmask = s->w_mask;
 | 
				
			||||||
 | 
					//;;; unsigned chain_length = s->max_chain_length;
 | 
				
			||||||
 | 
					//;;; if (s->prev_length >= s->good_match) {
 | 
				
			||||||
 | 
					//;;;     chain_length >>= 2;
 | 
				
			||||||
 | 
					//;;; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov edi, prev_length
 | 
				
			||||||
 | 
					        mov esi, good_match
 | 
				
			||||||
 | 
					        mov eax, WMask
 | 
				
			||||||
 | 
					        mov ebx, max_chain_length
 | 
				
			||||||
 | 
					        cmp edi, esi
 | 
				
			||||||
 | 
					        jl  LastMatchGood
 | 
				
			||||||
 | 
					        shr ebx, 2
 | 
				
			||||||
 | 
					LastMatchGood:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; chainlen is decremented once beforehand so that the function can
 | 
				
			||||||
 | 
					//;;; use the sign flag instead of the zero flag for the exit test.
 | 
				
			||||||
 | 
					//;;; It is then shifted into the high word, to make room for the wmask
 | 
				
			||||||
 | 
					//;;; value, which it will always accompany.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        dec ebx
 | 
				
			||||||
 | 
					        shl ebx, 16
 | 
				
			||||||
 | 
					        or  ebx, eax
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; on zlib only
 | 
				
			||||||
 | 
					//;;; if ((uInt)nice_match > s->lookahead) nice_match = s->lookahead;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov eax, nice_match
 | 
				
			||||||
 | 
					        mov [chainlenwmask], ebx
 | 
				
			||||||
 | 
					        mov r10d, Lookahead
 | 
				
			||||||
 | 
					        cmp r10d, eax
 | 
				
			||||||
 | 
					        cmovnl r10d, eax
 | 
				
			||||||
 | 
					        mov [nicematch],r10d
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; register Bytef *scan = s->window + s->strstart;
 | 
				
			||||||
 | 
					        mov r10, window_ad
 | 
				
			||||||
 | 
					        mov ebp, strstart
 | 
				
			||||||
 | 
					        lea r13, [r10 + rbp]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; Determine how many bytes the scan ptr is off from being
 | 
				
			||||||
 | 
					//;;; dword-aligned.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					         mov r9,r13
 | 
				
			||||||
 | 
					         neg r13
 | 
				
			||||||
 | 
					         and r13,3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; IPos limit = s->strstart > (IPos)MAX_DIST(s) ?
 | 
				
			||||||
 | 
					//;;;     s->strstart - (IPos)MAX_DIST(s) : NIL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov eax, window_size
 | 
				
			||||||
 | 
					        sub eax, MIN_LOOKAHEAD
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        xor edi,edi
 | 
				
			||||||
 | 
					        sub ebp, eax
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov r11d, prev_length
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        cmovng ebp,edi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; int best_len = s->prev_length;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; Store the sum of s->window + best_len in esi locally, and in esi.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					       lea  rsi,[r10+r11]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; register ush scan_start = *(ushf*)scan;
 | 
				
			||||||
 | 
					//;;; register ush scan_end   = *(ushf*)(scan+best_len-1);
 | 
				
			||||||
 | 
					//;;; Posf *prev = s->prev;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        movzx r12d,word ptr [r9]
 | 
				
			||||||
 | 
					        movzx ebx, word ptr [r9 + r11 - 1]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov rdi, prev_ad
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; Jump into the main loop.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov edx, [chainlenwmask]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        cmp bx,word ptr [rsi + r8 - 1]
 | 
				
			||||||
 | 
					        jz  LookupLoopIsZero
 | 
				
			||||||
 | 
									
 | 
				
			||||||
 | 
											
 | 
				
			||||||
 | 
											
 | 
				
			||||||
 | 
					LookupLoop1:
 | 
				
			||||||
 | 
					        and r8d, edx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        movzx   r8d, word ptr [rdi + r8*2]
 | 
				
			||||||
 | 
					        cmp r8d, ebp
 | 
				
			||||||
 | 
					        jbe LeaveNow
 | 
				
			||||||
 | 
							
 | 
				
			||||||
 | 
							
 | 
				
			||||||
 | 
							
 | 
				
			||||||
 | 
					        sub edx, 0x00010000
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        js  LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LoopEntry1:
 | 
				
			||||||
 | 
					        cmp bx,word ptr [rsi + r8 - 1]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jz  LookupLoopIsZero
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LookupLoop2:
 | 
				
			||||||
 | 
					        and r8d, edx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        movzx   r8d, word ptr [rdi + r8*2]
 | 
				
			||||||
 | 
					        cmp r8d, ebp
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jbe LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					        sub edx, 0x00010000
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        js  LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LoopEntry2:
 | 
				
			||||||
 | 
					        cmp bx,word ptr [rsi + r8 - 1]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jz  LookupLoopIsZero
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LookupLoop4:
 | 
				
			||||||
 | 
					        and r8d, edx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        movzx   r8d, word ptr [rdi + r8*2]
 | 
				
			||||||
 | 
					        cmp r8d, ebp
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jbe LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					        sub edx, 0x00010000
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        js  LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LoopEntry4:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        cmp bx,word ptr [rsi + r8 - 1]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jnz LookupLoop1
 | 
				
			||||||
 | 
					        jmp LookupLoopIsZero
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;;; do {
 | 
				
			||||||
 | 
					;;;     match = s->window + cur_match;
 | 
				
			||||||
 | 
					;;;     if (*(ushf*)(match+best_len-1) != scan_end ||
 | 
				
			||||||
 | 
					;;;         *(ushf*)match != scan_start) continue;
 | 
				
			||||||
 | 
					;;;     [...]
 | 
				
			||||||
 | 
					;;; } while ((cur_match = prev[cur_match & wmask]) > limit
 | 
				
			||||||
 | 
					;;;          && --chain_length != 0);
 | 
				
			||||||
 | 
					;;;
 | 
				
			||||||
 | 
					;;; Here is the inner loop of the function. The function will spend the
 | 
				
			||||||
 | 
					;;; majority of its time in this loop, and majority of that time will
 | 
				
			||||||
 | 
					;;; be spent in the first ten instructions.
 | 
				
			||||||
 | 
					;;;
 | 
				
			||||||
 | 
					;;; Within this loop:
 | 
				
			||||||
 | 
					;;; ebx = scanend
 | 
				
			||||||
 | 
					;;; r8d = curmatch
 | 
				
			||||||
 | 
					;;; edx = chainlenwmask - i.e., ((chainlen << 16) | wmask)
 | 
				
			||||||
 | 
					;;; esi = windowbestlen - i.e., (window + bestlen)
 | 
				
			||||||
 | 
					;;; edi = prev
 | 
				
			||||||
 | 
					;;; ebp = limit
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					.balign 16
 | 
				
			||||||
 | 
					LookupLoop:
 | 
				
			||||||
 | 
					        and r8d, edx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        movzx   r8d, word ptr [rdi + r8*2]
 | 
				
			||||||
 | 
					        cmp r8d, ebp
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jbe LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					        sub edx, 0x00010000
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        js  LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LoopEntry:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        cmp bx,word ptr [rsi + r8 - 1]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jnz LookupLoop1
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					LookupLoopIsZero:
 | 
				
			||||||
 | 
					        cmp     r12w, word ptr [r10 + r8]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jnz LookupLoop1
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; Store the current value of chainlen.
 | 
				
			||||||
 | 
					        mov [chainlenwmask], edx
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;;; Point edi to the string under scrutiny, and esi to the string we
 | 
				
			||||||
 | 
					;;; are hoping to match it up with. In actuality, esi and edi are
 | 
				
			||||||
 | 
					;;; both pointed (MAX_MATCH_8 - scanalign) bytes ahead, and edx is
 | 
				
			||||||
 | 
					;;; initialized to -(MAX_MATCH_8 - scanalign).
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					        lea rsi,[r8+r10]
 | 
				
			||||||
 | 
					        mov rdx, 0xfffffffffffffef8 //; -(MAX_MATCH_8)
 | 
				
			||||||
 | 
					        lea rsi, [rsi + r13 + 0x0108] //;MAX_MATCH_8]
 | 
				
			||||||
 | 
					        lea rdi, [r9 + r13 + 0x0108] //;MAX_MATCH_8]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        prefetcht1 [rsi+rdx]
 | 
				
			||||||
 | 
					        prefetcht1 [rdi+rdx]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;;; Test the strings for equality, 8 bytes at a time. At the end,
 | 
				
			||||||
 | 
					;;; adjust rdx so that it is offset to the exact byte that mismatched.
 | 
				
			||||||
 | 
					;;;
 | 
				
			||||||
 | 
					;;; We already know at this point that the first three bytes of the
 | 
				
			||||||
 | 
					;;; strings match each other, and they can be safely passed over before
 | 
				
			||||||
 | 
					;;; starting the compare loop. So what this code does is skip over 0-3
 | 
				
			||||||
 | 
					;;; bytes, as much as necessary in order to dword-align the edi
 | 
				
			||||||
 | 
					;;; pointer. (rsi will still be misaligned three times out of four.)
 | 
				
			||||||
 | 
					;;;
 | 
				
			||||||
 | 
					;;; It should be confessed that this loop usually does not represent
 | 
				
			||||||
 | 
					;;; much of the total running time. Replacing it with a more
 | 
				
			||||||
 | 
					;;; straightforward "rep cmpsb" would not drastically degrade
 | 
				
			||||||
 | 
					;;; performance.
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LoopCmps:
 | 
				
			||||||
 | 
					        mov rax, [rsi + rdx]
 | 
				
			||||||
 | 
					        xor rax, [rdi + rdx]
 | 
				
			||||||
 | 
					        jnz LeaveLoopCmps
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov rax, [rsi + rdx + 8]
 | 
				
			||||||
 | 
					        xor rax, [rdi + rdx + 8]
 | 
				
			||||||
 | 
					        jnz LeaveLoopCmps8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov rax, [rsi + rdx + 8+8]
 | 
				
			||||||
 | 
					        xor rax, [rdi + rdx + 8+8]
 | 
				
			||||||
 | 
					        jnz LeaveLoopCmps16
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        add rdx,8+8+8
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jnz  LoopCmps
 | 
				
			||||||
 | 
					        jmp  LenMaximum
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
							
 | 
				
			||||||
 | 
					LeaveLoopCmps16: add rdx,8
 | 
				
			||||||
 | 
					LeaveLoopCmps8: add rdx,8
 | 
				
			||||||
 | 
					LeaveLoopCmps:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        test    eax, 0x0000FFFF
 | 
				
			||||||
 | 
					        jnz LenLower
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        test eax,0xffffffff
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        jnz LenLower32
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        add rdx,4
 | 
				
			||||||
 | 
					        shr rax,32
 | 
				
			||||||
 | 
					        or ax,ax
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jnz LenLower
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LenLower32:
 | 
				
			||||||
 | 
					        shr eax,16
 | 
				
			||||||
 | 
					        add rdx,2
 | 
				
			||||||
 | 
							
 | 
				
			||||||
 | 
					LenLower:		
 | 
				
			||||||
 | 
					        sub al, 1
 | 
				
			||||||
 | 
					        adc rdx, 0
 | 
				
			||||||
 | 
					//;;; Calculate the length of the match. If it is longer than MAX_MATCH,
 | 
				
			||||||
 | 
					//;;; then automatically accept it as the best possible match and leave.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        lea rax, [rdi + rdx]
 | 
				
			||||||
 | 
					        sub rax, r9
 | 
				
			||||||
 | 
					        cmp eax, MAX_MATCH
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jge LenMaximum
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;;; If the length of the match is not longer than the best match we
 | 
				
			||||||
 | 
					;;; have so far, then forget it and return to the lookup loop.
 | 
				
			||||||
 | 
					;///////////////////////////////////
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					        cmp eax, r11d
 | 
				
			||||||
 | 
					        jg  LongerMatch
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        lea rsi,[r10+r11]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mov rdi, prev_ad
 | 
				
			||||||
 | 
					        mov edx, [chainlenwmask]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jmp LookupLoop
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					;;;         s->match_start = cur_match;
 | 
				
			||||||
 | 
					;;;         best_len = len;
 | 
				
			||||||
 | 
					;;;         if (len >= nice_match) break;
 | 
				
			||||||
 | 
					;;;         scan_end = *(ushf*)(scan+best_len-1);
 | 
				
			||||||
 | 
					*/
 | 
				
			||||||
 | 
					LongerMatch:
 | 
				
			||||||
 | 
					        mov r11d, eax
 | 
				
			||||||
 | 
					        mov match_start, r8d
 | 
				
			||||||
 | 
					        cmp eax, [nicematch]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jge LeaveNow
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        lea rsi,[r10+rax]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        movzx   ebx, word ptr [r9 + rax - 1]
 | 
				
			||||||
 | 
					        mov rdi, prev_ad
 | 
				
			||||||
 | 
					        mov edx, [chainlenwmask]
 | 
				
			||||||
 | 
							BEFORE_JMP
 | 
				
			||||||
 | 
					        jmp LookupLoop
 | 
				
			||||||
 | 
							AFTER_JMP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; Accept the current string, with the maximum possible length.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LenMaximum:
 | 
				
			||||||
 | 
					        mov r11d,MAX_MATCH
 | 
				
			||||||
 | 
					        mov match_start, r8d
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; if ((uInt)best_len <= s->lookahead) return (uInt)best_len;
 | 
				
			||||||
 | 
					//;;; return s->lookahead;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LeaveNow:
 | 
				
			||||||
 | 
					        mov eax, Lookahead
 | 
				
			||||||
 | 
					        cmp r11d, eax
 | 
				
			||||||
 | 
					        cmovng eax, r11d
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//;;; Restore the stack and return from whence we came.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					//        mov rsi,[save_rsi]
 | 
				
			||||||
 | 
					//        mov rdi,[save_rdi]
 | 
				
			||||||
 | 
					        mov rbx,[save_rbx]
 | 
				
			||||||
 | 
					        mov rbp,[save_rbp]
 | 
				
			||||||
 | 
					        mov r12,[save_r12]
 | 
				
			||||||
 | 
					        mov r13,[save_r13]
 | 
				
			||||||
 | 
					        mov r14,[save_r14]
 | 
				
			||||||
 | 
					        mov r15,[save_r15]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        ret 0
 | 
				
			||||||
 | 
					//; please don't remove this string !
 | 
				
			||||||
 | 
					//; Your can freely use gvmat64 in any free or commercial app
 | 
				
			||||||
 | 
					//; but it is far better don't remove the string in the binary!
 | 
				
			||||||
 | 
					 //   db     0dh,0ah,"asm686 with masm, optimised assembly code from Brian Raiter, written 1998, converted to amd 64 by Gilles Vollant 2005",0dh,0ah,0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					match_init:
 | 
				
			||||||
 | 
					  ret 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1 @@
 | 
				
			||||||
 | 
					See infback9.h for what this is and how to use it.
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,603 @@
 | 
				
			||||||
 | 
					/* infback9.c -- inflate deflate64 data using a call-back interface
 | 
				
			||||||
 | 
					 * Copyright (C) 1995-2008 Mark Adler
 | 
				
			||||||
 | 
					 * For conditions of distribution and use, see copyright notice in zlib.h
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "zutil.h"
 | 
				
			||||||
 | 
					#include "infback9.h"
 | 
				
			||||||
 | 
					#include "inftree9.h"
 | 
				
			||||||
 | 
					#include "inflate9.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#define WSIZE 65536UL
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					   strm provides memory allocation functions in zalloc and zfree, or
 | 
				
			||||||
 | 
					   Z_NULL to use the library memory allocation functions.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   window is a user-supplied window and output buffer that is 64K bytes.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					int ZEXPORT inflateBack9Init_(z_stream FAR *strm, unsigned char FAR *window,
 | 
				
			||||||
 | 
					                              const char *version, int stream_size) {
 | 
				
			||||||
 | 
					    struct inflate_state FAR *state;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (version == Z_NULL || version[0] != ZLIB_VERSION[0] ||
 | 
				
			||||||
 | 
					        stream_size != (int)(sizeof(z_stream)))
 | 
				
			||||||
 | 
					        return Z_VERSION_ERROR;
 | 
				
			||||||
 | 
					    if (strm == Z_NULL || window == Z_NULL)
 | 
				
			||||||
 | 
					        return Z_STREAM_ERROR;
 | 
				
			||||||
 | 
					    strm->msg = Z_NULL;                 /* in case we return an error */
 | 
				
			||||||
 | 
					    if (strm->zalloc == (alloc_func)0) {
 | 
				
			||||||
 | 
					        strm->zalloc = zcalloc;
 | 
				
			||||||
 | 
					        strm->opaque = (voidpf)0;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    if (strm->zfree == (free_func)0) strm->zfree = zcfree;
 | 
				
			||||||
 | 
					    state = (struct inflate_state FAR *)ZALLOC(strm, 1,
 | 
				
			||||||
 | 
					                                               sizeof(struct inflate_state));
 | 
				
			||||||
 | 
					    if (state == Z_NULL) return Z_MEM_ERROR;
 | 
				
			||||||
 | 
					    Tracev((stderr, "inflate: allocated\n"));
 | 
				
			||||||
 | 
					    strm->state = (voidpf)state;
 | 
				
			||||||
 | 
					    state->window = window;
 | 
				
			||||||
 | 
					    return Z_OK;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					   Build and output length and distance decoding tables for fixed code
 | 
				
			||||||
 | 
					   decoding.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					#ifdef MAKEFIXED
 | 
				
			||||||
 | 
					#include <stdio.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void makefixed9(void) {
 | 
				
			||||||
 | 
					    unsigned sym, bits, low, size;
 | 
				
			||||||
 | 
					    code *next, *lenfix, *distfix;
 | 
				
			||||||
 | 
					    struct inflate_state state;
 | 
				
			||||||
 | 
					    code fixed[544];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* literal/length table */
 | 
				
			||||||
 | 
					    sym = 0;
 | 
				
			||||||
 | 
					    while (sym < 144) state.lens[sym++] = 8;
 | 
				
			||||||
 | 
					    while (sym < 256) state.lens[sym++] = 9;
 | 
				
			||||||
 | 
					    while (sym < 280) state.lens[sym++] = 7;
 | 
				
			||||||
 | 
					    while (sym < 288) state.lens[sym++] = 8;
 | 
				
			||||||
 | 
					    next = fixed;
 | 
				
			||||||
 | 
					    lenfix = next;
 | 
				
			||||||
 | 
					    bits = 9;
 | 
				
			||||||
 | 
					    inflate_table9(LENS, state.lens, 288, &(next), &(bits), state.work);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* distance table */
 | 
				
			||||||
 | 
					    sym = 0;
 | 
				
			||||||
 | 
					    while (sym < 32) state.lens[sym++] = 5;
 | 
				
			||||||
 | 
					    distfix = next;
 | 
				
			||||||
 | 
					    bits = 5;
 | 
				
			||||||
 | 
					    inflate_table9(DISTS, state.lens, 32, &(next), &(bits), state.work);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* write tables */
 | 
				
			||||||
 | 
					    puts("    /* inffix9.h -- table for decoding deflate64 fixed codes");
 | 
				
			||||||
 | 
					    puts("     * Generated automatically by makefixed9().");
 | 
				
			||||||
 | 
					    puts("     */");
 | 
				
			||||||
 | 
					    puts("");
 | 
				
			||||||
 | 
					    puts("    /* WARNING: this file should *not* be used by applications.");
 | 
				
			||||||
 | 
					    puts("       It is part of the implementation of this library and is");
 | 
				
			||||||
 | 
					    puts("       subject to change. Applications should only use zlib.h.");
 | 
				
			||||||
 | 
					    puts("     */");
 | 
				
			||||||
 | 
					    puts("");
 | 
				
			||||||
 | 
					    size = 1U << 9;
 | 
				
			||||||
 | 
					    printf("    static const code lenfix[%u] = {", size);
 | 
				
			||||||
 | 
					    low = 0;
 | 
				
			||||||
 | 
					    for (;;) {
 | 
				
			||||||
 | 
					        if ((low % 6) == 0) printf("\n        ");
 | 
				
			||||||
 | 
					        printf("{%u,%u,%d}", lenfix[low].op, lenfix[low].bits,
 | 
				
			||||||
 | 
					               lenfix[low].val);
 | 
				
			||||||
 | 
					        if (++low == size) break;
 | 
				
			||||||
 | 
					        putchar(',');
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    puts("\n    };");
 | 
				
			||||||
 | 
					    size = 1U << 5;
 | 
				
			||||||
 | 
					    printf("\n    static const code distfix[%u] = {", size);
 | 
				
			||||||
 | 
					    low = 0;
 | 
				
			||||||
 | 
					    for (;;) {
 | 
				
			||||||
 | 
					        if ((low % 5) == 0) printf("\n        ");
 | 
				
			||||||
 | 
					        printf("{%u,%u,%d}", distfix[low].op, distfix[low].bits,
 | 
				
			||||||
 | 
					               distfix[low].val);
 | 
				
			||||||
 | 
					        if (++low == size) break;
 | 
				
			||||||
 | 
					        putchar(',');
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    puts("\n    };");
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#endif /* MAKEFIXED */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Macros for inflateBack(): */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Clear the input bit accumulator */
 | 
				
			||||||
 | 
					#define INITBITS() \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        hold = 0; \
 | 
				
			||||||
 | 
					        bits = 0; \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Assure that some input is available.  If input is requested, but denied,
 | 
				
			||||||
 | 
					   then return a Z_BUF_ERROR from inflateBack(). */
 | 
				
			||||||
 | 
					#define PULL() \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        if (have == 0) { \
 | 
				
			||||||
 | 
					            have = in(in_desc, &next); \
 | 
				
			||||||
 | 
					            if (have == 0) { \
 | 
				
			||||||
 | 
					                next = Z_NULL; \
 | 
				
			||||||
 | 
					                ret = Z_BUF_ERROR; \
 | 
				
			||||||
 | 
					                goto inf_leave; \
 | 
				
			||||||
 | 
					            } \
 | 
				
			||||||
 | 
					        } \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Get a byte of input into the bit accumulator, or return from inflateBack()
 | 
				
			||||||
 | 
					   with an error if there is no input available. */
 | 
				
			||||||
 | 
					#define PULLBYTE() \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        PULL(); \
 | 
				
			||||||
 | 
					        have--; \
 | 
				
			||||||
 | 
					        hold += (unsigned long)(*next++) << bits; \
 | 
				
			||||||
 | 
					        bits += 8; \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Assure that there are at least n bits in the bit accumulator.  If there is
 | 
				
			||||||
 | 
					   not enough available input to do that, then return from inflateBack() with
 | 
				
			||||||
 | 
					   an error. */
 | 
				
			||||||
 | 
					#define NEEDBITS(n) \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        while (bits < (unsigned)(n)) \
 | 
				
			||||||
 | 
					            PULLBYTE(); \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Return the low n bits of the bit accumulator (n <= 16) */
 | 
				
			||||||
 | 
					#define BITS(n) \
 | 
				
			||||||
 | 
					    ((unsigned)hold & ((1U << (n)) - 1))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Remove n bits from the bit accumulator */
 | 
				
			||||||
 | 
					#define DROPBITS(n) \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        hold >>= (n); \
 | 
				
			||||||
 | 
					        bits -= (unsigned)(n); \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Remove zero to seven bits as needed to go to a byte boundary */
 | 
				
			||||||
 | 
					#define BYTEBITS() \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        hold >>= bits & 7; \
 | 
				
			||||||
 | 
					        bits -= bits & 7; \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Assure that some output space is available, by writing out the window
 | 
				
			||||||
 | 
					   if it's full.  If the write fails, return from inflateBack() with a
 | 
				
			||||||
 | 
					   Z_BUF_ERROR. */
 | 
				
			||||||
 | 
					#define ROOM() \
 | 
				
			||||||
 | 
					    do { \
 | 
				
			||||||
 | 
					        if (left == 0) { \
 | 
				
			||||||
 | 
					            put = window; \
 | 
				
			||||||
 | 
					            left = WSIZE; \
 | 
				
			||||||
 | 
					            wrap = 1; \
 | 
				
			||||||
 | 
					            if (out(out_desc, put, (unsigned)left)) { \
 | 
				
			||||||
 | 
					                ret = Z_BUF_ERROR; \
 | 
				
			||||||
 | 
					                goto inf_leave; \
 | 
				
			||||||
 | 
					            } \
 | 
				
			||||||
 | 
					        } \
 | 
				
			||||||
 | 
					    } while (0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					   strm provides the memory allocation functions and window buffer on input,
 | 
				
			||||||
 | 
					   and provides information on the unused input on return.  For Z_DATA_ERROR
 | 
				
			||||||
 | 
					   returns, strm will also provide an error message.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   in() and out() are the call-back input and output functions.  When
 | 
				
			||||||
 | 
					   inflateBack() needs more input, it calls in().  When inflateBack() has
 | 
				
			||||||
 | 
					   filled the window with output, or when it completes with data in the
 | 
				
			||||||
 | 
					   window, it calls out() to write out the data.  The application must not
 | 
				
			||||||
 | 
					   change the provided input until in() is called again or inflateBack()
 | 
				
			||||||
 | 
					   returns.  The application must not change the window/output buffer until
 | 
				
			||||||
 | 
					   inflateBack() returns.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   in() and out() are called with a descriptor parameter provided in the
 | 
				
			||||||
 | 
					   inflateBack() call.  This parameter can be a structure that provides the
 | 
				
			||||||
 | 
					   information required to do the read or write, as well as accumulated
 | 
				
			||||||
 | 
					   information on the input and output such as totals and check values.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					   in() should return zero on failure.  out() should return non-zero on
 | 
				
			||||||
 | 
					   failure.  If either in() or out() fails, than inflateBack() returns a
 | 
				
			||||||
 | 
					   Z_BUF_ERROR.  strm->next_in can be checked for Z_NULL to see whether it
 | 
				
			||||||
 | 
					   was in() or out() that caused in the error.  Otherwise,  inflateBack()
 | 
				
			||||||
 | 
					   returns Z_STREAM_END on success, Z_DATA_ERROR for an deflate format
 | 
				
			||||||
 | 
					   error, or Z_MEM_ERROR if it could not allocate memory for the state.
 | 
				
			||||||
 | 
					   inflateBack() can also return Z_STREAM_ERROR if the input parameters
 | 
				
			||||||
 | 
					   are not correct, i.e. strm is Z_NULL or the state was not initialized.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					int ZEXPORT inflateBack9(z_stream FAR *strm, in_func in, void FAR *in_desc,
 | 
				
			||||||
 | 
					                         out_func out, void FAR *out_desc) {
 | 
				
			||||||
 | 
					    struct inflate_state FAR *state;
 | 
				
			||||||
 | 
					    z_const unsigned char FAR *next;    /* next input */
 | 
				
			||||||
 | 
					    unsigned char FAR *put;     /* next output */
 | 
				
			||||||
 | 
					    unsigned have;              /* available input */
 | 
				
			||||||
 | 
					    unsigned long left;         /* available output */
 | 
				
			||||||
 | 
					    inflate_mode mode;          /* current inflate mode */
 | 
				
			||||||
 | 
					    int lastblock;              /* true if processing last block */
 | 
				
			||||||
 | 
					    int wrap;                   /* true if the window has wrapped */
 | 
				
			||||||
 | 
					    unsigned char FAR *window;  /* allocated sliding window, if needed */
 | 
				
			||||||
 | 
					    unsigned long hold;         /* bit buffer */
 | 
				
			||||||
 | 
					    unsigned bits;              /* bits in bit buffer */
 | 
				
			||||||
 | 
					    unsigned extra;             /* extra bits needed */
 | 
				
			||||||
 | 
					    unsigned long length;       /* literal or length of data to copy */
 | 
				
			||||||
 | 
					    unsigned long offset;       /* distance back to copy string from */
 | 
				
			||||||
 | 
					    unsigned long copy;         /* number of stored or match bytes to copy */
 | 
				
			||||||
 | 
					    unsigned char FAR *from;    /* where to copy match bytes from */
 | 
				
			||||||
 | 
					    code const FAR *lencode;    /* starting table for length/literal codes */
 | 
				
			||||||
 | 
					    code const FAR *distcode;   /* starting table for distance codes */
 | 
				
			||||||
 | 
					    unsigned lenbits;           /* index bits for lencode */
 | 
				
			||||||
 | 
					    unsigned distbits;          /* index bits for distcode */
 | 
				
			||||||
 | 
					    code here;                  /* current decoding table entry */
 | 
				
			||||||
 | 
					    code last;                  /* parent table entry */
 | 
				
			||||||
 | 
					    unsigned len;               /* length to copy for repeats, bits to drop */
 | 
				
			||||||
 | 
					    int ret;                    /* return code */
 | 
				
			||||||
 | 
					    static const unsigned short order[19] = /* permutation of code lengths */
 | 
				
			||||||
 | 
					        {16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15};
 | 
				
			||||||
 | 
					#include "inffix9.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* Check that the strm exists and that the state was initialized */
 | 
				
			||||||
 | 
					    if (strm == Z_NULL || strm->state == Z_NULL)
 | 
				
			||||||
 | 
					        return Z_STREAM_ERROR;
 | 
				
			||||||
 | 
					    state = (struct inflate_state FAR *)strm->state;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* Reset the state */
 | 
				
			||||||
 | 
					    strm->msg = Z_NULL;
 | 
				
			||||||
 | 
					    mode = TYPE;
 | 
				
			||||||
 | 
					    lastblock = 0;
 | 
				
			||||||
 | 
					    wrap = 0;
 | 
				
			||||||
 | 
					    window = state->window;
 | 
				
			||||||
 | 
					    next = strm->next_in;
 | 
				
			||||||
 | 
					    have = next != Z_NULL ? strm->avail_in : 0;
 | 
				
			||||||
 | 
					    hold = 0;
 | 
				
			||||||
 | 
					    bits = 0;
 | 
				
			||||||
 | 
					    put = window;
 | 
				
			||||||
 | 
					    left = WSIZE;
 | 
				
			||||||
 | 
					    lencode = Z_NULL;
 | 
				
			||||||
 | 
					    distcode = Z_NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* Inflate until end of block marked as last */
 | 
				
			||||||
 | 
					    for (;;)
 | 
				
			||||||
 | 
					        switch (mode) {
 | 
				
			||||||
 | 
					        case TYPE:
 | 
				
			||||||
 | 
					            /* determine and dispatch block type */
 | 
				
			||||||
 | 
					            if (lastblock) {
 | 
				
			||||||
 | 
					                BYTEBITS();
 | 
				
			||||||
 | 
					                mode = DONE;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            NEEDBITS(3);
 | 
				
			||||||
 | 
					            lastblock = BITS(1);
 | 
				
			||||||
 | 
					            DROPBITS(1);
 | 
				
			||||||
 | 
					            switch (BITS(2)) {
 | 
				
			||||||
 | 
					            case 0:                             /* stored block */
 | 
				
			||||||
 | 
					                Tracev((stderr, "inflate:     stored block%s\n",
 | 
				
			||||||
 | 
					                        lastblock ? " (last)" : ""));
 | 
				
			||||||
 | 
					                mode = STORED;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            case 1:                             /* fixed block */
 | 
				
			||||||
 | 
					                lencode = lenfix;
 | 
				
			||||||
 | 
					                lenbits = 9;
 | 
				
			||||||
 | 
					                distcode = distfix;
 | 
				
			||||||
 | 
					                distbits = 5;
 | 
				
			||||||
 | 
					                Tracev((stderr, "inflate:     fixed codes block%s\n",
 | 
				
			||||||
 | 
					                        lastblock ? " (last)" : ""));
 | 
				
			||||||
 | 
					                mode = LEN;                     /* decode codes */
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            case 2:                             /* dynamic block */
 | 
				
			||||||
 | 
					                Tracev((stderr, "inflate:     dynamic codes block%s\n",
 | 
				
			||||||
 | 
					                        lastblock ? " (last)" : ""));
 | 
				
			||||||
 | 
					                mode = TABLE;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            case 3:
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid block type";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            DROPBITS(2);
 | 
				
			||||||
 | 
					            break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case STORED:
 | 
				
			||||||
 | 
					            /* get and verify stored block length */
 | 
				
			||||||
 | 
					            BYTEBITS();                         /* go to byte boundary */
 | 
				
			||||||
 | 
					            NEEDBITS(32);
 | 
				
			||||||
 | 
					            if ((hold & 0xffff) != ((hold >> 16) ^ 0xffff)) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid stored block lengths";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            length = (unsigned)hold & 0xffff;
 | 
				
			||||||
 | 
					            Tracev((stderr, "inflate:       stored length %lu\n",
 | 
				
			||||||
 | 
					                    length));
 | 
				
			||||||
 | 
					            INITBITS();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* copy stored block from input to output */
 | 
				
			||||||
 | 
					            while (length != 0) {
 | 
				
			||||||
 | 
					                copy = length;
 | 
				
			||||||
 | 
					                PULL();
 | 
				
			||||||
 | 
					                ROOM();
 | 
				
			||||||
 | 
					                if (copy > have) copy = have;
 | 
				
			||||||
 | 
					                if (copy > left) copy = left;
 | 
				
			||||||
 | 
					                zmemcpy(put, next, copy);
 | 
				
			||||||
 | 
					                have -= copy;
 | 
				
			||||||
 | 
					                next += copy;
 | 
				
			||||||
 | 
					                left -= copy;
 | 
				
			||||||
 | 
					                put += copy;
 | 
				
			||||||
 | 
					                length -= copy;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            Tracev((stderr, "inflate:       stored end\n"));
 | 
				
			||||||
 | 
					            mode = TYPE;
 | 
				
			||||||
 | 
					            break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case TABLE:
 | 
				
			||||||
 | 
					            /* get dynamic table entries descriptor */
 | 
				
			||||||
 | 
					            NEEDBITS(14);
 | 
				
			||||||
 | 
					            state->nlen = BITS(5) + 257;
 | 
				
			||||||
 | 
					            DROPBITS(5);
 | 
				
			||||||
 | 
					            state->ndist = BITS(5) + 1;
 | 
				
			||||||
 | 
					            DROPBITS(5);
 | 
				
			||||||
 | 
					            state->ncode = BITS(4) + 4;
 | 
				
			||||||
 | 
					            DROPBITS(4);
 | 
				
			||||||
 | 
					            if (state->nlen > 286) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"too many length symbols";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            Tracev((stderr, "inflate:       table sizes ok\n"));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* get code length code lengths (not a typo) */
 | 
				
			||||||
 | 
					            state->have = 0;
 | 
				
			||||||
 | 
					            while (state->have < state->ncode) {
 | 
				
			||||||
 | 
					                NEEDBITS(3);
 | 
				
			||||||
 | 
					                state->lens[order[state->have++]] = (unsigned short)BITS(3);
 | 
				
			||||||
 | 
					                DROPBITS(3);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            while (state->have < 19)
 | 
				
			||||||
 | 
					                state->lens[order[state->have++]] = 0;
 | 
				
			||||||
 | 
					            state->next = state->codes;
 | 
				
			||||||
 | 
					            lencode = (code const FAR *)(state->next);
 | 
				
			||||||
 | 
					            lenbits = 7;
 | 
				
			||||||
 | 
					            ret = inflate_table9(CODES, state->lens, 19, &(state->next),
 | 
				
			||||||
 | 
					                                &(lenbits), state->work);
 | 
				
			||||||
 | 
					            if (ret) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid code lengths set";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            Tracev((stderr, "inflate:       code lengths ok\n"));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* get length and distance code code lengths */
 | 
				
			||||||
 | 
					            state->have = 0;
 | 
				
			||||||
 | 
					            while (state->have < state->nlen + state->ndist) {
 | 
				
			||||||
 | 
					                for (;;) {
 | 
				
			||||||
 | 
					                    here = lencode[BITS(lenbits)];
 | 
				
			||||||
 | 
					                    if ((unsigned)(here.bits) <= bits) break;
 | 
				
			||||||
 | 
					                    PULLBYTE();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                if (here.val < 16) {
 | 
				
			||||||
 | 
					                    NEEDBITS(here.bits);
 | 
				
			||||||
 | 
					                    DROPBITS(here.bits);
 | 
				
			||||||
 | 
					                    state->lens[state->have++] = here.val;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else {
 | 
				
			||||||
 | 
					                    if (here.val == 16) {
 | 
				
			||||||
 | 
					                        NEEDBITS(here.bits + 2);
 | 
				
			||||||
 | 
					                        DROPBITS(here.bits);
 | 
				
			||||||
 | 
					                        if (state->have == 0) {
 | 
				
			||||||
 | 
					                            strm->msg = (char *)"invalid bit length repeat";
 | 
				
			||||||
 | 
					                            mode = BAD;
 | 
				
			||||||
 | 
					                            break;
 | 
				
			||||||
 | 
					                        }
 | 
				
			||||||
 | 
					                        len = (unsigned)(state->lens[state->have - 1]);
 | 
				
			||||||
 | 
					                        copy = 3 + BITS(2);
 | 
				
			||||||
 | 
					                        DROPBITS(2);
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                    else if (here.val == 17) {
 | 
				
			||||||
 | 
					                        NEEDBITS(here.bits + 3);
 | 
				
			||||||
 | 
					                        DROPBITS(here.bits);
 | 
				
			||||||
 | 
					                        len = 0;
 | 
				
			||||||
 | 
					                        copy = 3 + BITS(3);
 | 
				
			||||||
 | 
					                        DROPBITS(3);
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                    else {
 | 
				
			||||||
 | 
					                        NEEDBITS(here.bits + 7);
 | 
				
			||||||
 | 
					                        DROPBITS(here.bits);
 | 
				
			||||||
 | 
					                        len = 0;
 | 
				
			||||||
 | 
					                        copy = 11 + BITS(7);
 | 
				
			||||||
 | 
					                        DROPBITS(7);
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                    if (state->have + copy > state->nlen + state->ndist) {
 | 
				
			||||||
 | 
					                        strm->msg = (char *)"invalid bit length repeat";
 | 
				
			||||||
 | 
					                        mode = BAD;
 | 
				
			||||||
 | 
					                        break;
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                    while (copy--)
 | 
				
			||||||
 | 
					                        state->lens[state->have++] = (unsigned short)len;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* handle error breaks in while */
 | 
				
			||||||
 | 
					            if (mode == BAD) break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* check for end-of-block code (better have one) */
 | 
				
			||||||
 | 
					            if (state->lens[256] == 0) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid code -- missing end-of-block";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* build code tables -- note: do not change the lenbits or distbits
 | 
				
			||||||
 | 
					               values here (9 and 6) without reading the comments in inftree9.h
 | 
				
			||||||
 | 
					               concerning the ENOUGH constants, which depend on those values */
 | 
				
			||||||
 | 
					            state->next = state->codes;
 | 
				
			||||||
 | 
					            lencode = (code const FAR *)(state->next);
 | 
				
			||||||
 | 
					            lenbits = 9;
 | 
				
			||||||
 | 
					            ret = inflate_table9(LENS, state->lens, state->nlen,
 | 
				
			||||||
 | 
					                            &(state->next), &(lenbits), state->work);
 | 
				
			||||||
 | 
					            if (ret) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid literal/lengths set";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            distcode = (code const FAR *)(state->next);
 | 
				
			||||||
 | 
					            distbits = 6;
 | 
				
			||||||
 | 
					            ret = inflate_table9(DISTS, state->lens + state->nlen,
 | 
				
			||||||
 | 
					                            state->ndist, &(state->next), &(distbits),
 | 
				
			||||||
 | 
					                            state->work);
 | 
				
			||||||
 | 
					            if (ret) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid distances set";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            Tracev((stderr, "inflate:       codes ok\n"));
 | 
				
			||||||
 | 
					            mode = LEN;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case LEN:
 | 
				
			||||||
 | 
					            /* get a literal, length, or end-of-block code */
 | 
				
			||||||
 | 
					            for (;;) {
 | 
				
			||||||
 | 
					                here = lencode[BITS(lenbits)];
 | 
				
			||||||
 | 
					                if ((unsigned)(here.bits) <= bits) break;
 | 
				
			||||||
 | 
					                PULLBYTE();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            if (here.op && (here.op & 0xf0) == 0) {
 | 
				
			||||||
 | 
					                last = here;
 | 
				
			||||||
 | 
					                for (;;) {
 | 
				
			||||||
 | 
					                    here = lencode[last.val +
 | 
				
			||||||
 | 
					                            (BITS(last.bits + last.op) >> last.bits)];
 | 
				
			||||||
 | 
					                    if ((unsigned)(last.bits + here.bits) <= bits) break;
 | 
				
			||||||
 | 
					                    PULLBYTE();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                DROPBITS(last.bits);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            DROPBITS(here.bits);
 | 
				
			||||||
 | 
					            length = (unsigned)here.val;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* process literal */
 | 
				
			||||||
 | 
					            if (here.op == 0) {
 | 
				
			||||||
 | 
					                Tracevv((stderr, here.val >= 0x20 && here.val < 0x7f ?
 | 
				
			||||||
 | 
					                        "inflate:         literal '%c'\n" :
 | 
				
			||||||
 | 
					                        "inflate:         literal 0x%02x\n", here.val));
 | 
				
			||||||
 | 
					                ROOM();
 | 
				
			||||||
 | 
					                *put++ = (unsigned char)(length);
 | 
				
			||||||
 | 
					                left--;
 | 
				
			||||||
 | 
					                mode = LEN;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* process end of block */
 | 
				
			||||||
 | 
					            if (here.op & 32) {
 | 
				
			||||||
 | 
					                Tracevv((stderr, "inflate:         end of block\n"));
 | 
				
			||||||
 | 
					                mode = TYPE;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* invalid code */
 | 
				
			||||||
 | 
					            if (here.op & 64) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid literal/length code";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* length code -- get extra bits, if any */
 | 
				
			||||||
 | 
					            extra = (unsigned)(here.op) & 31;
 | 
				
			||||||
 | 
					            if (extra != 0) {
 | 
				
			||||||
 | 
					                NEEDBITS(extra);
 | 
				
			||||||
 | 
					                length += BITS(extra);
 | 
				
			||||||
 | 
					                DROPBITS(extra);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            Tracevv((stderr, "inflate:         length %lu\n", length));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* get distance code */
 | 
				
			||||||
 | 
					            for (;;) {
 | 
				
			||||||
 | 
					                here = distcode[BITS(distbits)];
 | 
				
			||||||
 | 
					                if ((unsigned)(here.bits) <= bits) break;
 | 
				
			||||||
 | 
					                PULLBYTE();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            if ((here.op & 0xf0) == 0) {
 | 
				
			||||||
 | 
					                last = here;
 | 
				
			||||||
 | 
					                for (;;) {
 | 
				
			||||||
 | 
					                    here = distcode[last.val +
 | 
				
			||||||
 | 
					                            (BITS(last.bits + last.op) >> last.bits)];
 | 
				
			||||||
 | 
					                    if ((unsigned)(last.bits + here.bits) <= bits) break;
 | 
				
			||||||
 | 
					                    PULLBYTE();
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                DROPBITS(last.bits);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            DROPBITS(here.bits);
 | 
				
			||||||
 | 
					            if (here.op & 64) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid distance code";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            offset = (unsigned)here.val;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* get distance extra bits, if any */
 | 
				
			||||||
 | 
					            extra = (unsigned)(here.op) & 15;
 | 
				
			||||||
 | 
					            if (extra != 0) {
 | 
				
			||||||
 | 
					                NEEDBITS(extra);
 | 
				
			||||||
 | 
					                offset += BITS(extra);
 | 
				
			||||||
 | 
					                DROPBITS(extra);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            if (offset > WSIZE - (wrap ? 0: left)) {
 | 
				
			||||||
 | 
					                strm->msg = (char *)"invalid distance too far back";
 | 
				
			||||||
 | 
					                mode = BAD;
 | 
				
			||||||
 | 
					                break;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            Tracevv((stderr, "inflate:         distance %lu\n", offset));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            /* copy match from window to output */
 | 
				
			||||||
 | 
					            do {
 | 
				
			||||||
 | 
					                ROOM();
 | 
				
			||||||
 | 
					                copy = WSIZE - offset;
 | 
				
			||||||
 | 
					                if (copy < left) {
 | 
				
			||||||
 | 
					                    from = put + copy;
 | 
				
			||||||
 | 
					                    copy = left - copy;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                else {
 | 
				
			||||||
 | 
					                    from = put - offset;
 | 
				
			||||||
 | 
					                    copy = left;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					                if (copy > length) copy = length;
 | 
				
			||||||
 | 
					                length -= copy;
 | 
				
			||||||
 | 
					                left -= copy;
 | 
				
			||||||
 | 
					                do {
 | 
				
			||||||
 | 
					                    *put++ = *from++;
 | 
				
			||||||
 | 
					                } while (--copy);
 | 
				
			||||||
 | 
					            } while (length != 0);
 | 
				
			||||||
 | 
					            break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case DONE:
 | 
				
			||||||
 | 
					            /* inflate stream terminated properly -- write leftover output */
 | 
				
			||||||
 | 
					            ret = Z_STREAM_END;
 | 
				
			||||||
 | 
					            if (left < WSIZE) {
 | 
				
			||||||
 | 
					                if (out(out_desc, window, (unsigned)(WSIZE - left)))
 | 
				
			||||||
 | 
					                    ret = Z_BUF_ERROR;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            goto inf_leave;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        case BAD:
 | 
				
			||||||
 | 
					            ret = Z_DATA_ERROR;
 | 
				
			||||||
 | 
					            goto inf_leave;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        default:                /* can't happen, but makes compilers happy */
 | 
				
			||||||
 | 
					            ret = Z_STREAM_ERROR;
 | 
				
			||||||
 | 
					            goto inf_leave;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* Return unused input */
 | 
				
			||||||
 | 
					  inf_leave:
 | 
				
			||||||
 | 
					    strm->next_in = next;
 | 
				
			||||||
 | 
					    strm->avail_in = have;
 | 
				
			||||||
 | 
					    return ret;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int ZEXPORT inflateBack9End(z_stream FAR *strm) {
 | 
				
			||||||
 | 
					    if (strm == Z_NULL || strm->state == Z_NULL || strm->zfree == (free_func)0)
 | 
				
			||||||
 | 
					        return Z_STREAM_ERROR;
 | 
				
			||||||
 | 
					    ZFREE(strm, strm->state);
 | 
				
			||||||
 | 
					    strm->state = Z_NULL;
 | 
				
			||||||
 | 
					    Tracev((stderr, "inflate: end\n"));
 | 
				
			||||||
 | 
					    return Z_OK;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,37 @@
 | 
				
			||||||
 | 
					/* infback9.h -- header for using inflateBack9 functions
 | 
				
			||||||
 | 
					 * Copyright (C) 2003 Mark Adler
 | 
				
			||||||
 | 
					 * For conditions of distribution and use, see copyright notice in zlib.h
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * This header file and associated patches provide a decoder for PKWare's
 | 
				
			||||||
 | 
					 * undocumented deflate64 compression method (method 9).  Use with infback9.c,
 | 
				
			||||||
 | 
					 * inftree9.h, inftree9.c, and inffix9.h.  These patches are not supported.
 | 
				
			||||||
 | 
					 * This should be compiled with zlib, since it uses zutil.h and zutil.o.
 | 
				
			||||||
 | 
					 * This code has not yet been tested on 16-bit architectures.  See the
 | 
				
			||||||
 | 
					 * comments in zlib.h for inflateBack() usage.  These functions are used
 | 
				
			||||||
 | 
					 * identically, except that there is no windowBits parameter, and a 64K
 | 
				
			||||||
 | 
					 * window must be provided.  Also if int's are 16 bits, then a zero for
 | 
				
			||||||
 | 
					 * the third parameter of the "out" function actually means 65536UL.
 | 
				
			||||||
 | 
					 * zlib.h must be included before this header file.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef __cplusplus
 | 
				
			||||||
 | 
					extern "C" {
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ZEXTERN int ZEXPORT inflateBack9(z_stream FAR *strm,
 | 
				
			||||||
 | 
					                                 in_func in, void FAR *in_desc,
 | 
				
			||||||
 | 
					                                 out_func out, void FAR *out_desc);
 | 
				
			||||||
 | 
					ZEXTERN int ZEXPORT inflateBack9End(z_stream FAR *strm);
 | 
				
			||||||
 | 
					ZEXTERN int ZEXPORT inflateBack9Init_(z_stream FAR *strm,
 | 
				
			||||||
 | 
					                                      unsigned char FAR *window,
 | 
				
			||||||
 | 
					                                      const char *version,
 | 
				
			||||||
 | 
					                                      int stream_size);
 | 
				
			||||||
 | 
					#define inflateBack9Init(strm, window) \
 | 
				
			||||||
 | 
					        inflateBack9Init_((strm), (window), \
 | 
				
			||||||
 | 
					        ZLIB_VERSION, sizeof(z_stream))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef __cplusplus
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue