| 
									
										
										
										
											2016-09-28 14:40:37 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2018-02-07 14:13:13 +00:00
										 |  |  |     Copyright (c) 2016-2018 Chung, Hyung-Hwan. All rights reserved. | 
					
						
							| 
									
										
										
										
											2016-09-28 14:40:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |     modification, are permitted provided that the following conditions | 
					
						
							|  |  |  |     are met: | 
					
						
							|  |  |  |     1. Redistributions of source code must retain the above copyright | 
					
						
							|  |  |  |        notice, this list of conditions and the following disclaimer. | 
					
						
							|  |  |  |     2. Redistributions in binary form must reproduce the above copyright | 
					
						
							|  |  |  |        notice, this list of conditions and the following disclaimer in the | 
					
						
							|  |  |  |        documentation and/or other materials provided with the distribution. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR | 
					
						
							|  |  |  |     IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | 
					
						
							|  |  |  |     OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. | 
					
						
							|  |  |  |     IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, | 
					
						
							|  |  |  |     INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | 
					
						
							|  |  |  |     NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
					
						
							|  |  |  |     DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
					
						
							|  |  |  |     THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
					
						
							|  |  |  |     (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 
					
						
							|  |  |  |     THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | #include <hak-utl.h>
 | 
					
						
							|  |  |  | #include <hak-str.h>
 | 
					
						
							|  |  |  | #include <hak.h>
 | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* some naming conventions
 | 
					
						
							|  |  |  |  *  bchars, uchars -> pointer and length | 
					
						
							|  |  |  |  *  bcstr, ucstr -> null-terminated string pointer | 
					
						
							|  |  |  |  *  btouchars -> bchars to uchars | 
					
						
							|  |  |  |  *  utobchars -> uchars to bchars | 
					
						
							|  |  |  |  *  btoucstr -> bcstr to ucstr | 
					
						
							|  |  |  |  *  utobcstr -> ucstr to bcstr | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-09-28 14:40:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_oow_t hak_hash_bytes_ (const hak_oob_t* ptr, hak_oow_t len) | 
					
						
							| 
									
										
										
										
											2016-09-28 14:40:37 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_oow_t hv; | 
					
						
							|  |  |  | 	HAK_HASH_BYTES (hv, ptr, len); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	/* constrain the hash value to be representable in a small integer
 | 
					
						
							|  |  |  | 	 * for convenience sake */ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	return hv % ((hak_oow_t)HAK_SMOOI_MAX + 1); | 
					
						
							| 
									
										
										
										
											2016-09-28 14:40:37 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | /* ----------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | int hak_convbtouchars (hak_t* hak, const hak_bch_t* bcs, hak_oow_t* bcslen, hak_uch_t* ucs, hak_oow_t* ucslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* length bound */ | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	n = hak_conv_bchars_to_uchars_with_cmgr(bcs, bcslen, ucs, ucslen, HAK_CMGR(hak), 0); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (n <= -1) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		/* -1: illegal character, -2: buffer too small, -3: incomplete sequence */ | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 		hak_seterrnum(hak, (n == -2)? HAK_EBUFFULL: HAK_EECERR); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | int hak_convutobchars (hak_t* hak, const hak_uch_t* ucs, hak_oow_t* ucslen, hak_bch_t* bcs, hak_oow_t* bcslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* length bound */ | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	n = hak_conv_uchars_to_bchars_with_cmgr(ucs, ucslen, bcs, bcslen, HAK_CMGR(hak)); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (n <= -1) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 		hak_seterrnum(hak, (n == -2)? HAK_EBUFFULL: HAK_EECERR); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | int hak_convbtoucstr (hak_t* hak, const hak_bch_t* bcs, hak_oow_t* bcslen, hak_uch_t* ucs, hak_oow_t* ucslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* null-terminated. */ | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	n = hak_conv_bcstr_to_ucstr_with_cmgr(bcs, bcslen, ucs, ucslen, HAK_CMGR(hak), 0); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (n <= -1) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 		hak_seterrnum(hak, (n == -2)? HAK_EBUFFULL: HAK_EECERR); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | int hak_convutobcstr (hak_t* hak, const hak_uch_t* ucs, hak_oow_t* ucslen, hak_bch_t* bcs, hak_oow_t* bcslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* null-terminated */ | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	n = hak_conv_ucstr_to_bcstr_with_cmgr(ucs, ucslen, bcs, bcslen, HAK_CMGR(hak)); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (n <= -1) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 		hak_seterrnum(hak, (n == -2)? HAK_EBUFFULL: HAK_EECERR); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ----------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_uch_t* hak_dupbtoucharswithheadroom (hak_t* hak, hak_oow_t headroom_bytes, const hak_bch_t* bcs, hak_oow_t bcslen, hak_oow_t* ucslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_oow_t inlen, outlen; | 
					
						
							|  |  |  | 	hak_uch_t* ptr; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	inlen = bcslen; | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	if (hak_convbtouchars(hak, bcs, &inlen, HAK_NULL, &outlen) <= -1) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* note it's also an error if no full conversion is made in this function */ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_uch_t*)hak_allocmem(hak, headroom_bytes + ((outlen + 1) * HAK_SIZEOF(hak_uch_t))); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	inlen = bcslen; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_uch_t*)((hak_oob_t*)ptr + headroom_bytes); | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	hak_convbtouchars(hak, bcs, &inlen, ptr, &outlen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	/* hak_convbtouchars() doesn't null-terminate the target.
 | 
					
						
							|  |  |  | 	 * but in hak_dupbtouchars(), i allocate space. so i don't mind | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	 * null-terminating it with 1 extra character overhead */ | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | 	ptr[outlen] = '\0'; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	if (ucslen) *ucslen = outlen; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_uch_t* hak_dupbtouchars (hak_t* hak, const hak_bch_t* bcs, hak_oow_t bcslen, hak_oow_t* ucslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	return hak_dupbtoucharswithheadroom(hak, 0, bcs, bcslen, ucslen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_bch_t* hak_duputobcharswithheadroom (hak_t* hak, hak_oow_t headroom_bytes, const hak_uch_t* ucs, hak_oow_t ucslen, hak_oow_t* bcslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_oow_t inlen, outlen; | 
					
						
							|  |  |  | 	hak_bch_t* ptr; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	inlen = ucslen; | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	if (hak_convutobchars(hak, ucs, &inlen, HAK_NULL, &outlen) <= -1) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* note it's also an error if no full conversion is made in this function */ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_bch_t*)hak_allocmem(hak, headroom_bytes + ((outlen + 1) * HAK_SIZEOF(hak_bch_t))); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	inlen = ucslen; | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_bch_t*)((hak_oob_t*)ptr + headroom_bytes); | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	hak_convutobchars(hak, ucs, &inlen, ptr, &outlen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ptr[outlen] = '\0'; | 
					
						
							|  |  |  | 	if (bcslen) *bcslen = outlen; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_bch_t* hak_duputobchars (hak_t* hak, const hak_uch_t* ucs, hak_oow_t ucslen, hak_oow_t* bcslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	return hak_duputobcharswithheadroom(hak, 0, ucs, ucslen, bcslen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ----------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_uch_t* hak_dupbtoucstrwithheadroom (hak_t* hak, hak_oow_t headroom_bytes, const hak_bch_t* bcs, hak_oow_t* ucslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_oow_t inlen, outlen; | 
					
						
							|  |  |  | 	hak_uch_t* ptr; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	if (hak_convbtoucstr(hak, bcs, &inlen, HAK_NULL, &outlen) <= -1) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* note it's also an error if no full conversion is made in this function */ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	outlen++; | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_uch_t*)hak_allocmem(hak, headroom_bytes + (outlen * HAK_SIZEOF(hak_uch_t))); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	hak_convbtoucstr(hak, bcs, &inlen, ptr, &outlen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	if (ucslen) *ucslen = outlen; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_uch_t* hak_dupbtoucstr (hak_t* hak, const hak_bch_t* bcs, hak_oow_t* ucslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	return hak_dupbtoucstrwithheadroom(hak, 0, bcs, ucslen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_bch_t* hak_duputobcstrwithheadroom (hak_t* hak, hak_oow_t headroom_bytes, const hak_uch_t* ucs, hak_oow_t* bcslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_oow_t inlen, outlen; | 
					
						
							|  |  |  | 	hak_bch_t* ptr; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	if (hak_convutobcstr(hak, ucs, &inlen, HAK_NULL, &outlen) <= -1) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* note it's also an error if no full conversion is made in this function */ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	outlen++; | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_bch_t*)hak_allocmem(hak, headroom_bytes + (outlen * HAK_SIZEOF(hak_bch_t))); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_bch_t*)((hak_oob_t*)ptr + headroom_bytes); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	hak_convutobcstr(hak, ucs, &inlen, ptr, &outlen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	if (bcslen) *bcslen = outlen; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_bch_t* hak_duputobcstr (hak_t* hak, const hak_uch_t* ucs, hak_oow_t* bcslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	return hak_duputobcstrwithheadroom(hak, 0, ucs, bcslen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | } | 
					
						
							|  |  |  | /* ----------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_uch_t* hak_dupuchars (hak_t* hak, const hak_uch_t* ucs, hak_oow_t ucslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_uch_t* ptr; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_uch_t*)hak_allocmem(hak, (ucslen + 1) * HAK_SIZEOF(hak_uch_t)); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_copy_uchars (ptr, ucs, ucslen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	ptr[ucslen] = '\0'; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_bch_t* hak_dupbchars (hak_t* hak, const hak_bch_t* bcs, hak_oow_t bcslen) | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_bch_t* ptr; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	ptr = (hak_bch_t*)hak_allocmem(hak, (bcslen + 1) * HAK_SIZEOF(hak_bch_t)); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_copy_bchars (ptr, bcs, bcslen); | 
					
						
							| 
									
										
										
										
											2018-02-05 10:43:25 +00:00
										 |  |  | 	ptr[bcslen] = '\0'; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_uch_t* hak_dupucstr (hak_t* hak, const hak_uch_t* ucs, hak_oow_t* ucslen) | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_oow_t len; | 
					
						
							|  |  |  | 	hak_uch_t* ptr; | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	len = hak_count_ucstr(ucs); | 
					
						
							|  |  |  | 	ptr = hak_dupuchars(hak, ucs, len); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (ucslen) *ucslen = len; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | hak_bch_t* hak_dupbcstr (hak_t* hak, const hak_bch_t* bcs, hak_oow_t* bcslen) | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_oow_t len; | 
					
						
							|  |  |  | 	hak_bch_t* ptr; | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	len = hak_count_bcstr(bcs); | 
					
						
							|  |  |  | 	ptr = hak_dupbchars(hak, bcs, len); | 
					
						
							|  |  |  | 	if (HAK_UNLIKELY(!ptr)) return HAK_NULL; | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (bcslen) *bcslen = len; | 
					
						
							|  |  |  | 	return ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | /* ----------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | void hak_add_ntime (hak_ntime_t* z, const hak_ntime_t* x, const hak_ntime_t* y) | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_ntime_sec_t xs, ys; | 
					
						
							|  |  |  | 	hak_ntime_nsec_t ns; | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	/*HAK_ASSERT(x->nsec >= 0 && x->nsec < HAK_NSECS_PER_SEC);
 | 
					
						
							|  |  |  | 	HAK_ASSERT(y->nsec >= 0 && y->nsec < HAK_NSECS_PER_SEC);*/ | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ns = x->nsec + y->nsec; | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	if (ns >= HAK_NSECS_PER_SEC) | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		ns = ns - HAK_NSECS_PER_SEC; | 
					
						
							|  |  |  | 		if (x->sec == HAK_TYPE_MAX(hak_ntime_sec_t)) | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			if (y->sec >= 0) goto overflow; | 
					
						
							|  |  |  | 			xs = x->sec; | 
					
						
							|  |  |  | 			ys = y->sec + 1; /* this won't overflow */ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			xs = x->sec + 1; /* this won't overflow */ | 
					
						
							|  |  |  | 			ys = y->sec; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		xs = x->sec; | 
					
						
							|  |  |  | 		ys = y->sec; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	if ((ys >= 1 && xs > HAK_TYPE_MAX(hak_ntime_sec_t) - ys) || | 
					
						
							|  |  |  | 	    (ys <= -1 && xs < HAK_TYPE_MIN(hak_ntime_sec_t) - ys)) | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		if (xs >= 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		overflow: | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 			xs = HAK_TYPE_MAX(hak_ntime_sec_t); | 
					
						
							|  |  |  | 			ns = HAK_NSECS_PER_SEC - 1; | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 			xs = HAK_TYPE_MIN(hak_ntime_sec_t); | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 			ns = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		xs = xs + ys; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	z->sec = xs; | 
					
						
							|  |  |  | 	z->nsec = ns; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | void hak_sub_ntime (hak_ntime_t* z, const hak_ntime_t* x, const hak_ntime_t* y) | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	hak_ntime_sec_t xs, ys; | 
					
						
							|  |  |  | 	hak_ntime_nsec_t ns; | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-05 10:52:02 +09:00
										 |  |  | 	/*HAK_ASSERT(x->nsec >= 0 && x->nsec < HAK_NSECS_PER_SEC);
 | 
					
						
							|  |  |  | 	HAK_ASSERT(y->nsec >= 0 && y->nsec < HAK_NSECS_PER_SEC);*/ | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ns = x->nsec - y->nsec; | 
					
						
							|  |  |  | 	if (ns < 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		ns = ns + HAK_NSECS_PER_SEC; | 
					
						
							|  |  |  | 		if (x->sec == HAK_TYPE_MIN(hak_ntime_sec_t)) | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			if (y->sec <= 0) goto underflow; | 
					
						
							|  |  |  | 			xs = x->sec; | 
					
						
							|  |  |  | 			ys = y->sec - 1; /* this won't underflow */ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			xs = x->sec - 1; /* this won't underflow */ | 
					
						
							|  |  |  | 			ys = y->sec; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		xs = x->sec; | 
					
						
							|  |  |  | 		ys = y->sec; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	if ((ys >= 1 && xs < HAK_TYPE_MIN(hak_ntime_sec_t) + ys) || | 
					
						
							|  |  |  | 	    (ys <= -1 && xs > HAK_TYPE_MAX(hak_ntime_sec_t) + ys)) | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		if (xs >= 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 			xs = HAK_TYPE_MAX(hak_ntime_sec_t); | 
					
						
							|  |  |  | 			ns = HAK_NSECS_PER_SEC - 1; | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		underflow: | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 			xs = HAK_TYPE_MIN(hak_ntime_sec_t); | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 			ns = 0; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2021-03-28 18:12:07 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-09-03 06:21:01 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		xs = xs - ys; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	z->sec = xs; | 
					
						
							|  |  |  | 	z->nsec = ns; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | /* ----------------------------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | const hak_bch_t* hak_get_base_name_from_bcstr_path (const hak_bch_t* path) | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	const hak_bch_t* p, * last = HAK_NULL; | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (p = path; *p != '\0'; p++) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		if (HAK_IS_PATH_SEP(*p)) last = p; | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	return (last == HAK_NULL)? path: (last + 1); | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | const hak_uch_t* hak_get_base_name_from_ucstr_path (const hak_uch_t* path) | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	const hak_uch_t* p, * last = HAK_NULL; | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (p = path; *p != '\0'; p++) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 		if (HAK_IS_PATH_SEP(*p)) last = p; | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-02 23:58:15 +09:00
										 |  |  | 	return (last == HAK_NULL)? path: (last + 1); | 
					
						
							| 
									
										
										
										
											2024-04-27 21:16:00 +09:00
										 |  |  | } |