mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-25 18:38:00 +03:00 
			
		
		
		
	bmove_allign -> bmove_align Added OLAP function ROLLUP Split mysql_fix_privilege_tables to a script and a .sql data file Added new (MEMROOT*) functions to avoid calling current_thd() when creating some common objects. Added table_alias_charset, for easier --lower-case-table-name handling Better SQL_MODE handling (Setting complex options also sets sub options) New (faster) assembler string functions for x86 BitKeeper/etc/ignore: added libmysqld/sql_state.c client/mysql.cc: Added SQLSTATE to error messages Added new function put_error() to be able to clean up some old code. client/mysqltest.c: Write ERROR SQLSTATE for all errors dbug/dbug.c: Portability fixes include/m_string.h: Rename bmove_allign as bmove_align include/mysql.h: Added SQLSTATE (for embedded version) include/mysql_com.h: Send correct SQLSTATE for the error to the client libmysql/libmysql.c: Changed default error state to HY000 Applied code cleanup patch libmysqld/Makefile.am: Added sql_state.cc libmysqld/libmysqld.c: Added sqlstate mysql-test/r/analyse.result: Updated results mysql-test/r/ansi.result: Updated results mysql-test/r/auto_increment.result: Updated results mysql-test/r/bdb-deadlock.result: Updated results mysql-test/r/bdb.result: Updated results mysql-test/r/comments.result: Updated results mysql-test/r/create.result: Updated results mysql-test/r/ctype_collate.result: Updated results mysql-test/r/delayed.result: Updated results mysql-test/r/delete.result: Updated results mysql-test/r/derived.result: Updated results mysql-test/r/distinct.result: Updated results mysql-test/r/drop.result: Updated results mysql-test/r/err000001.result: Updated results mysql-test/r/explain.result: Updated results mysql-test/r/flush.result: Updated results mysql-test/r/fulltext.result: Updated results mysql-test/r/func_gconcat.result: Updated results mysql-test/r/func_system.result: Updated results mysql-test/r/grant_cache.result: Updated results mysql-test/r/group_by.result: Updated results mysql-test/r/handler.result: Updated results mysql-test/r/heap.result: Updated results mysql-test/r/heap_btree.result: Updated results mysql-test/r/heap_hash.result: Updated results mysql-test/r/innodb.result: Updated results mysql-test/r/innodb_handler.result: Updated results mysql-test/r/insert_select.result: Updated results mysql-test/r/insert_update.result: Updated results mysql-test/r/join.result: Updated results mysql-test/r/join_outer.result: Updated results mysql-test/r/key.result: Updated results mysql-test/r/lock.result: Updated results mysql-test/r/lock_multi.result: Updated results mysql-test/r/merge.result: Updated results mysql-test/r/multi_update.result: Updated results mysql-test/r/myisam.result: Updated results mysql-test/r/null.result: Updated results mysql-test/r/olap.result: Updated results mysql-test/r/order_by.result: Updated results mysql-test/r/packet.result: Updated results mysql-test/r/query_cache.result: Updated results mysql-test/r/row.result: Updated results mysql-test/r/rpl000001.result: Updated results mysql-test/r/rpl000009.result: Updated results mysql-test/r/rpl_empty_master_crash.result: Updated results mysql-test/r/rpl_log.result: Updated results mysql-test/r/rpl_replicate_do.result: Updated results mysql-test/r/rpl_rotate_logs.result: Updated results mysql-test/r/select.result: Updated results mysql-test/r/select_safe.result: Updated results mysql-test/r/show_check.result: Updated results mysql-test/r/sql_mode.result: Updated results mysql-test/r/subselect.result: Updated results mysql-test/r/temp_table.result: Updated results mysql-test/r/truncate.result: Updated results mysql-test/r/type_blob.result: Updated results mysql-test/r/type_decimal.result: Updated results mysql-test/r/type_float.result: Updated results mysql-test/r/type_ranges.result: Updated results mysql-test/r/union.result: Updated results mysql-test/r/update.result: Updated results mysql-test/r/user_var.result: Updated results mysql-test/r/varbinary.result: Updated results mysql-test/r/variables.result: Updated results mysql-test/t/ansi.test: Test of sql_mode mysql-test/t/derived.test: Updated results mysql-test/t/func_system.test: Make this independen of the MySQL server name mysql-test/t/lowercase_table.test: Cleanup mysql-test/t/olap.test: A lot of new tests mysql-test/t/sql_mode.test: More test for sql_mode mysql-test/t/subselect.test: Added a few new tests (to find a bug in the item_ref code) scripts/Makefile.am: Added mysql_fix_privilege_tables.sql scripts/mysql_fix_privilege_tables.sh: Totally new script. This bascily just pipes mysql_fix_privilege_tables.sql through 'mysql' to 'mysqld' sql/Makefile.am: Added sql_state.cc sql/item.cc: Extended Item_field::eq() to be able to better match GROUP BY fields on the command line. Needed for ROLLUP sql/item.h: Added function to be able to avoid calling current_thd() when doing new Item. sql/item_sum.cc: Moved copy_or_same() and some reset() functions from item_sum.h Needed to be able to access thd->mem_root. sql/item_sum.h: Moved some functions to item_sum.cc Added make_unique() for ROLLUP sql/item_uniq.h: Fixed return value sql/mysql_priv.h: Updated MODE flags sql/mysqld.cc: Added ANSI as it's own mode Moved charset_info variables here Cleaned up handler_count handling (for NT) Added table_alias_charset, for easier --lower-case-table-name handling sql/net_serv.cc: New comment sql/protocol.cc: Send SQLSTATE to client sql/set_var.cc: Better SQL_MODE handling (Setting complex options also sets sub options) sql/set_var.h: Better SQL_MODE handling sql/sql_base.cc: Make alias depend on --lower-case-table-names Make find_item_in_list also check database name sql/sql_cache.cc: Indentation cleanup sql/sql_list.h: Added safety assert Addes support of alloc without current_thd() sql/sql_prepare.cc: Update after prototype change sql/sql_select.cc: Added ROLLUP sql/sql_select.h: structures for rollup sql/sql_show.cc: Easier SQL_MODE handling sql/sql_string.cc: Move CHARSET_INFO to mysqld (to be together with all other global variables) sql/sql_string.h: Added function to be able to avoid calling current_thd() when doing new Item. sql/sql_table.cc: Simpler --lower-case-table-name handling sql/sql_union.cc: Update after prototype change sql/sql_yacc.yy: ROLLUP sql/unireg.h: bmove_allign ->bmove_align strings/Makefile.am: Fix to be able to compile str_test.c strings/ctype.c: Removed empty lines strings/str_test.c: Added test of bmove_align strings/strings-x86.s: Faster bmove_align, bmove_upp and strmake strings/strings.asm: move_allg
		
			
				
	
	
		
			1061 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			NASM
		
	
	
	
	
	
			
		
		
	
	
			1061 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			NASM
		
	
	
	
	
	
| ; Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
 | |
| ; 
 | |
| ; This library is free software; you can redistribute it and/or
 | |
| ; modify it under the terms of the GNU Library 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
 | |
| ; Library General Public License for more details.
 | |
| ; 
 | |
| ; You should have received a copy of the GNU Library 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
 | |
| 
 | |
| ; Note that if you don't have a macro assembler (like MASM) to compile
 | |
| ; this file, you can instead compile all *.c files in the string
 | |
| ; directory.
 | |
| 
 | |
| 	TITLE   Stringfunctions that we use often at MSDOS / Intel 8086
 | |
| 
 | |
| ifndef M_I386
 | |
| 	.8087
 | |
| 	DOSSEG
 | |
| 	.MODEL LARGE
 | |
| 	.CODE
 | |
| 
 | |
| 	;
 | |
| 	; Some macros
 | |
| 	;
 | |
| 
 | |
| q_movs	MACRO				; as rep movsb but quicker
 | |
| 	shr	cx,1
 | |
| 	rep	movsw			; Move 2 bytes at a time
 | |
| 	adc	cx,cx
 | |
| 	rep	movsb			; Move last byte if any
 | |
| 	ENDM
 | |
| 
 | |
| q_stos	MACRO				; as rep stosb but quicker
 | |
| 	mov	ah,al			; For word store
 | |
| 	shr	cx,1
 | |
| 	rep	stosw			; Move 2 bytes at a time
 | |
| 	adc	cx,cx
 | |
| 	rep	stosb			; Move last byte if any
 | |
|  	ENDM
 | |
| 
 | |
| ifndef  ZTC				; If not using ZORTECH compiler
 | |
| 	;
 | |
| 	; Compare memory
 | |
| 	; Args: s1,s2,length
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_bcmp
 | |
| _bcmp	PROC
 | |
| 	mov	bx,bp			; Save bp
 | |
| 	mov	dx,di			; Save di
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	push	si
 | |
| 	les	di,DWORD PTR [bp+8]	; s2
 | |
| 	lds	si,DWORD PTR [bp+4]	; s1
 | |
| 	mov	cx,WORD PTR [bp+12]	; Length of memory-area
 | |
| 	jcxz	@F			; Length = 0, return same
 | |
| ;	cld				; Work uppward
 | |
| 	repe	cmpsb			; Compare strings
 | |
| 	jz	@F			; Match found
 | |
| 	inc	cx			; return matchpoint +1
 | |
| @@:	mov	ax,cx			; Return 0 if match, else pos from end
 | |
| 	pop	si
 | |
| 	pop	ds
 | |
| 	mov	di,dx
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _bcmp	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Find a char in a string
 | |
| 	; Arg: str,char
 | |
| 	; Ret: pointer to found char or NullS
 | |
| 	;
 | |
| 
 | |
| ifdef better_stringfunctions		; Breaks window linkage (broken linking)
 | |
| 
 | |
| 	PUBLIC	_strchr
 | |
| _strchr	PROC
 | |
| 	mov	bx,bp			; Save bp and di
 | |
| 	mov	dx,di
 | |
| 	mov	bp,sp
 | |
| 	les	di,DWORD PTR [bp+4]	; str
 | |
| 	mov	ah,BYTE PTR [bp+8]	; search
 | |
| 	xor	al,al			; for scasb to find end
 | |
| 
 | |
| @@:	cmp	ah,es:[di]
 | |
| 	jz	@F			; Found char
 | |
| 	scasb
 | |
| 	jnz	@B			; Not end
 | |
| 	xor	di,di			; Not found
 | |
| 	mov	es,di
 | |
| @@:	mov	ax,di
 | |
| 	mov	di,dx			; Restore
 | |
| 	mov	dx,es			; Seg adr
 | |
| 	mov	bp,bx			; Restore
 | |
| 	ret
 | |
| _strchr	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Find length of string
 | |
| 	; arg: str
 | |
| 	; ret: length
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strlen
 | |
| _strlen	PROC
 | |
| 	mov	bx,sp
 | |
| 	mov	dx,di
 | |
| 	les	di,DWORD PTR ss:[bx+4]	; Str
 | |
| 	xor	al,al			; Find end of string
 | |
| 	mov	cx,-1
 | |
| ;	cld
 | |
| 	repne	scasb			; Find strend or length
 | |
| 	inc	cx			; Calc strlength
 | |
| 	not	cx
 | |
| 	mov	ax,cx
 | |
| 	mov	di,dx			; Restore register
 | |
| 	ret
 | |
| _strlen	ENDP
 | |
| 
 | |
| endif
 | |
| 
 | |
| 	;
 | |
| 	; Move a string
 | |
| 	; arg: dst,src
 | |
| 	; ret: end-null of to
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strmov
 | |
| _strmov	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	cx,si
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	push	di
 | |
| 	les	di,DWORD PTR [bp+4]	; dst
 | |
| 	lds	si,DWORD PTR [bp+8]	; src
 | |
| ;	cld
 | |
| @@:	mov	al,ds:[si]
 | |
| 	movsb				; move arg
 | |
| 	and	al,al
 | |
| 	jnz	@B			; Not last
 | |
| 	lea	ax,WORD PTR [di-1]	; Set DX:AX to point at last null
 | |
| 	mov	dx,es
 | |
| 	pop	di
 | |
| 	pop	ds
 | |
| 	mov	si,cx
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _strmov	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Fill a area of memory with a walue
 | |
| 	; Args: to,length,fillchar
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_bfill
 | |
| _bfill	PROC
 | |
| 	mov	bx,sp			; Get args through BX
 | |
| 	mov	al,BYTE PTR ss:[bx+10]	; Fill
 | |
| bfill_10:
 | |
| 	mov	dx,di			; Save di
 | |
| 	les	di,DWORD PTR ss:[bx+4]	; Memory pointer
 | |
| 	mov	cx,WORD PTR ss:[bx+8]	; Length
 | |
| ;	cld
 | |
| 	q_stos
 | |
| 	mov	di,dx
 | |
| 	ret
 | |
| _bfill	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Fill a area with null
 | |
| 	; Args: to,length
 | |
| 
 | |
| 	PUBLIC	_bzero
 | |
| _bzero	PROC
 | |
| 	mov	bx,sp			; Get args through BX
 | |
| 	mov	al,0			; Fill with null
 | |
| 	jmp	short bfill_10
 | |
| _bzero	ENDP
 | |
| 
 | |
| endif	; ZTC
 | |
| 
 | |
| 	;
 | |
| 	; Move a memory area
 | |
| 	; Args: to,from,length
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_bmove
 | |
| _bmove	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	dx,di
 | |
| 	mov	ax,si
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	lds	si,DWORD PTR [bp+8]	; from
 | |
| 	les	di,DWORD PTR [bp+4]	; to
 | |
| 	mov	cx,WORD PTR [bp+12]	; Length of memory-area
 | |
| ;	cld				; Work uppward
 | |
| 	rep	movsb			; Not q_movs because overlap ?
 | |
| 	pop	ds
 | |
| 	mov	si,ax
 | |
| 	mov	di,dx
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _bmove	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Move a alligned, not overlapped, by (long) divided memory area
 | |
| 	; Args: to,from,length
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_bmove_align
 | |
| _bmove_align	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	dx,di
 | |
| 	mov	ax,si
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	lds	si,DWORD PTR [bp+8]	; from
 | |
| 	les	di,DWORD PTR [bp+4]	; to
 | |
| 	mov	cx,WORD PTR [bp+12]	; Length of memory-area
 | |
| ;	cld				; Work uppward
 | |
| 	inc	cx			; fix if not divisible with word
 | |
| 	shr	cx,1
 | |
| 	rep	movsw			; Move 2 bytes at a time
 | |
| 	pop	ds
 | |
| 	mov	si,ax
 | |
| 	mov	di,dx
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _bmove_align	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Move a string from higher to lower
 | |
| 	; Arg from+1,to+1,length
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_bmove_upp
 | |
| _bmove_upp	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	dx,di
 | |
| 	mov	ax,si
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	lds	si,DWORD PTR [bp+8]	; from
 | |
| 	les	di,DWORD PTR [bp+4]	; to
 | |
| 	mov	cx,WORD PTR [bp+12]	; Length of memory-area
 | |
| 	dec	di			; Don't move last arg
 | |
| 	dec	si
 | |
| 	std				; Work downward
 | |
| 	rep	movsb			; Not q_movs because overlap ?
 | |
| 	cld				; C compilator want cld
 | |
| 	pop	ds
 | |
| 	mov	si,ax
 | |
| 	mov	di,dx
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _bmove_upp ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Append fillchars to string
 | |
| 	; Args: dest,len,fill
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strappend
 | |
| _strappend	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	dx,di
 | |
| 	mov	bp,sp
 | |
| 	les	di,DWORD PTR [bp+4]	; Memory pointer
 | |
| 	mov	cx,WORD PTR [bp+8]	; Length
 | |
| 	sub	al,al			; Find end of string
 | |
| ;	cld
 | |
| 	repne	scasb
 | |
| 	jnz	sa_99			; String to long, shorten it
 | |
| 	mov	al,BYTE PTR [bp+10]	; Fillchar
 | |
| 	dec	di			; Point at end null
 | |
| 	inc	cx			; rep made one dec for null-char
 | |
| 	q_stos				; Store al in string
 | |
| sa_99:	mov	BYTE PTR es:[di],0	; End of string
 | |
| 	mov	di,dx
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _strappend	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Find if string contains any char in another string
 | |
| 	; Arg: str,set
 | |
| 	; Ret: Pointer to first found char in str
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strcont
 | |
| _strcont	PROC
 | |
| 	mov	bx,bp			; Save bp and di in regs
 | |
| 	mov	dx,di
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	push	si
 | |
| 	lds	si,DWORD PTR [bp+4]	; str
 | |
| 	les	di,DWORD PTR [bp+8]	; Set
 | |
| 	mov	cx,di			; Save for loop
 | |
| 	xor	ah,ah			; For endtest
 | |
| 	jmp	sc_60
 | |
| 
 | |
| sc_10:	scasb
 | |
| 	jz	sc_fo			; Found char
 | |
| sc_20:	cmp	ah,es:[di]		; Test if null
 | |
| 	jnz	sc_10			; Not end of set yet
 | |
| 	inc	si			; Next char in str
 | |
| 	mov	di,cx			; es:di = Set
 | |
| sc_60:	mov	al,ds:[si]		; Test if this char exist
 | |
| 	and	al,al
 | |
| 	jnz	sc_20			; Not end of string
 | |
| 	sub	si,si			; Return Null
 | |
| 	mov	ds,si
 | |
| sc_fo:	mov	ax,si			; Char found here
 | |
| 	mov	di,dx			; Restore
 | |
| 	mov	dx,ds			; Seg of found char
 | |
| 	pop	si
 | |
| 	pop	ds
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _strcont	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Found end of string
 | |
| 	; Arg: str
 | |
| 	; ret: Pointer to end null
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strend
 | |
| _strend	PROC
 | |
| 	mov	bx,sp
 | |
| 	mov	dx,di			; Save
 | |
| 	les	di,DWORD PTR ss:[bx+4]	; str
 | |
| 	mov	cx,-1
 | |
| 	sub	al,al			; Find end of string
 | |
| ;	cld
 | |
| 	repne	scasb
 | |
| 	lea	ax,WORD PTR [di-1]	; Endpos i DX:AX
 | |
| 	mov	di,dx			; Restore
 | |
| 	mov	dx,es
 | |
| 	ret
 | |
| _strend	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Make a string with len fill-chars and endnull
 | |
| 	; Args: dest,len,fill
 | |
| 	; Ret:  dest+len
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strfill
 | |
| _strfill	PROC
 | |
| 	mov	bx,bp			; Save sp
 | |
| 	mov	bp,sp
 | |
| 	push	di
 | |
| 	les	di,DWORD PTR [bp+4]	; Memory pointer
 | |
| 	mov	cx,WORD PTR [bp+8]	; Length
 | |
| 	mov	al,BYTE PTR [bp+10]	; Fill
 | |
| ;	cld
 | |
| 	q_stos
 | |
| 	mov	BYTE PTR es:[di],0	; End NULL
 | |
| 	mov	ax,di			; End i DX:AX
 | |
| 	mov	dx,es
 | |
| 	pop	di
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _strfill	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Find a char in or end of a string
 | |
| 	; Arg: str,char
 | |
| 	; Ret: pointer to found char or NullS
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strcend
 | |
| _strcend	PROC
 | |
| 	mov	bx,bp			; Save bp and di
 | |
| 	mov	dx,di
 | |
| 	mov	bp,sp
 | |
| 	les	di,DWORD PTR [bp+4]	; str
 | |
| 	mov	ah,BYTE PTR [bp+8]	; search
 | |
| 	xor	al,al			; for scasb to find end
 | |
| 
 | |
| @@:	cmp	ah,es:[di]
 | |
| 	jz	@F			; Found char
 | |
| 	scasb
 | |
| 	jnz	@B			; Not end
 | |
| 	dec 	di			; Not found, point at end of string
 | |
| @@:	mov	ax,di
 | |
| 	mov	di,dx			; Restore
 | |
| 	mov	dx,es			; Seg adr
 | |
| 	mov	bp,bx			; Restore
 | |
| 	ret
 | |
| _strcend	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Test if string has a given suffix
 | |
| 	;
 | |
| 
 | |
| PUBLIC  _is_prefix
 | |
| _is_prefix PROC
 | |
| 	mov	dx,di			; Save di
 | |
| 	mov	bx,sp			; Arguments through bx
 | |
| 	push	ds
 | |
| 	push	si
 | |
| 	les	di,DWORD PTR ss:[bx+8]	; s2
 | |
| 	lds	si,DWORD PTR ss:[bx+4]	; s1
 | |
| 	mov	ax,1			; Ok and zero-test
 | |
| ;	cld				; Work uppward
 | |
| @@:	cmp	ah,es:[di]
 | |
| 	jz	suf_ok			; End of string; found suffix
 | |
| 	cmpsb				; Compare strings
 | |
| 	jz	@B			; Same, possible prefix
 | |
| 	xor	ax,ax			; Not suffix
 | |
| suf_ok:	pop	si
 | |
| 	pop	ds
 | |
| 	mov	di,dx
 | |
| 	ret
 | |
| _is_prefix ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Find a substring in string
 | |
| 	; Arg: str,search
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strstr
 | |
| _strstr	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	push	di
 | |
| 	push	si
 | |
| 	lds	si,DWORD PTR [bp+4]	; str
 | |
| 	les	di,DWORD PTR [bp+8]	; search
 | |
| 	mov	cx,di
 | |
| 	inc	cx			; CX = search+1
 | |
| 	mov	ah,es:[di]		; AH = First char in search
 | |
| 	jmp	sf_10
 | |
| 
 | |
| sf_00:	mov	si,dx			; si = Current str-pos
 | |
| sf_10:	mov	al,ds:[si]		; Test if this char exist
 | |
| 	and	al,al
 | |
| 	jz	sf_90			; End of string, didn't find search
 | |
| 	inc	si
 | |
| 	cmp	al,ah
 | |
| 	jnz	sf_10			; Didn't find first char, continue
 | |
| 	mov	dx,si			; Save str-pos in DX
 | |
| 	mov	di,cx
 | |
| sf_20:	cmp	BYTE PTR es:[di],0
 | |
| 	jz	sf_fo			; Found substring
 | |
| 	cmpsb
 | |
| 	jz	sf_20			; Char ok
 | |
| 	jmp	sf_00			; Next str-pos
 | |
| 
 | |
| sf_90:	sub	dx,dx			; Return Null
 | |
| 	mov	ds,dx
 | |
| 	inc	dx			; Because of following dec
 | |
| sf_fo:	mov	ax,dx			; Char found here
 | |
| 	dec	ax			; Pointed one after
 | |
| 	mov	dx,ds
 | |
| 	pop	si
 | |
| 	pop	di			; End
 | |
| 	pop	ds
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _strstr	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Find a substring in string, return index
 | |
| 	; Arg: str,search
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strinstr
 | |
| _strinstr	PROC
 | |
| 	push	bp
 | |
| 	mov	bp,sp
 | |
| 	push	di
 | |
| 	les	di,DWORD PTR [bp+10]	; search
 | |
| 	push	es
 | |
| 	push	di
 | |
| 	les	di,DWORD PTR [bp+6]	; str
 | |
| 	push	es
 | |
| 	push	di
 | |
| 	call	_strstr
 | |
| 	mov	cx,ax
 | |
| 	or	cx,dx
 | |
| 	jz	si_99
 | |
| 	sub	ax,di			; Pos from start
 | |
| 	inc	ax			; And first pos = 1
 | |
| si_99:	add	sp,8
 | |
| 	pop	di
 | |
| 	pop	bp
 | |
| 	ret
 | |
| _strinstr	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Make a string of len length from another string
 | |
| 	; Arg: dst,src,length
 | |
| 	; ret: end of dst
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strmake
 | |
| _strmake	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	push	di
 | |
| 	push	si
 | |
| 	les	di,DWORD PTR [bp+4]	; dst
 | |
| 	lds	si,DWORD PTR [bp+8]	; src
 | |
| 	mov	cx,WORD PTR [bp+12]	; Length of memory-area
 | |
| 	xor	al,al			; For test of end-null
 | |
| 	jcxz	sm_90			; Nothing to move, put zero at end.
 | |
| ;	cld				; Work uppward
 | |
| 
 | |
| @@:	cmp	al,ds:[si]		; Next char to move
 | |
| 	movsb				; move arg
 | |
| 	jz	sm_99			; last char, we are ready
 | |
| 	loop	@B			; Continue moving
 | |
| sm_90:	mov	BYTE PTR es:[di],al	; Set end pos
 | |
| 	inc	di			; Fix that di points at end null
 | |
| sm_99:	dec	di			; di points now at end null
 | |
| 	mov	ax,di			; Ret value in DX:AX
 | |
| 	mov	dx,es
 | |
| 	pop	si
 | |
| 	pop	di
 | |
| 	pop	ds
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _strmake	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Find length of string with maxlength
 | |
| 	; arg: str,maxlength
 | |
| 	; ret: length
 | |
| 	;
 | |
| 
 | |
| 	PUBLIC	_strnlen
 | |
| _strnlen	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	bp,sp
 | |
| 	push	di
 | |
| 	les	di,DWORD PTR [bp+4]	; Str
 | |
| 	mov	cx,WORD PTR [bp+8]	; length
 | |
| 	mov	dx,di			; Save str to calc length
 | |
| 	jcxz	sn_10			; Length = 0
 | |
| 	xor	al,al			; Find end of string
 | |
| ;	cld
 | |
| 	repne	scasb			; Find strend or length
 | |
| 	jnz	sn_10
 | |
| 	dec	di			; DI points at last null
 | |
| sn_10:	mov	ax,di
 | |
| 	sub	ax,dx			; Ax = length
 | |
| 	pop	di
 | |
| 	mov	bp,bx
 | |
| 	ret
 | |
| _strnlen	ENDP
 | |
| 
 | |
| 	;
 | |
| 	; Move a string with max len chars
 | |
| 	; arg: dst,src,len
 | |
| 	; ret: pos to first null or dst+len
 | |
| 
 | |
| 	PUBLIC	_strnmov
 | |
| _strnmov	PROC
 | |
| 	mov	bx,bp
 | |
| 	mov	bp,sp
 | |
| 	push	ds
 | |
| 	push	di
 | |
| 	push	si
 | |
| 	les	di,DWORD PTR [bp+4]	; dst
 | |
| 	lds	si,DWORD PTR [bp+8]	; src
 | |
| 	mov	cx,WORD PTR [bp+12]	; length
 | |
| 	jcxz	snm_99			; Nothing to do
 | |
| 	xor	al,al			; For test of end-null
 | |
| ;	cld
 | |
| 
 | |
| @@:	cmp	al,ds:[si]		; Next char to move
 | |
| 	movsb				; move arg
 | |
| 	jz	snm_20			; last char, fill with null
 | |
| 	loop	@B			; Continue moving
 | |
| 	inc	di			; Point two after last
 | |
| snm_20:	dec	di			; Point at first null (or last+1)
 | |
| snm_99:	mov	ax,di			; Pointer at last char
 | |
| 	mov	dx,es			; To-segment
 | |
| 	pop	si
 | |
| 	pop	di
 | |
| 	pop	ds
 | |
| 	mov	bp,bx			; Restore
 | |
| 	ret
 | |
| _strnmov	ENDP
 | |
| 
 | |
| else	; M_I386
 | |
| 
 | |
| include macros.asm
 | |
| 
 | |
| q_stos	MACRO				; as rep stosb but quicker, Uses edx
 | |
| 	mov	ah,al			;(2) Set up a 32 bit pattern.
 | |
| 	mov	edx,eax			;(2)
 | |
| 	shl	edx,16			;(3)
 | |
| 	or	eax,edx			;(2) EAX has the 32 bit pattern.
 | |
| 
 | |
| 	mov	edx,ecx			;(2) Save the count of bytes.
 | |
| 	shr	ecx,2			;(2) Number of dwords.
 | |
| 	rep	stosd			;(5 + 5n)
 | |
| 	mov	cl,3			;(2)
 | |
| 	and	ecx,edx			;(2) Fill in the remaining odd bytes.
 | |
| 	rep	stosb			; Move last bytes if any
 | |
| 	ENDM
 | |
| 
 | |
| fix_es	MACRO	fix_cld			; Load ES if neaded
 | |
|   ife ESeqDS
 | |
| 	mov	ax,ds
 | |
| 	mov	es,ax
 | |
|   endif
 | |
|   ifnb <fix_cld>
 | |
| 	cld
 | |
|   endif
 | |
| 	ENDM
 | |
| 
 | |
| 	;
 | |
| 	; Move a memory area
 | |
| 	; Args: to,from,length
 | |
| 	; Acts as one byte was moved a-time from dst to source.
 | |
| 	;
 | |
| 
 | |
| 	begcode bmove
 | |
| 	public	_bmove
 | |
| _bmove	proc near
 | |
| 	fix_es	1
 | |
| 	mov	edx,edi
 | |
| 	mov	eax,esi
 | |
| 	mov	edi,P-SIZEPTR[esp]	;p1
 | |
| 	mov	esi,P[esp]		;p2
 | |
| 	mov	ecx,P+SIZEPTR[esp]
 | |
| 	rep	movsb			; Not q_movs because overlap ?
 | |
| 	mov	esi,eax
 | |
| 	mov	edi,edx
 | |
| 	ret
 | |
| _bmove	ENDP
 | |
| 	endcode bmove
 | |
| 
 | |
| 	;
 | |
| 	; Move a alligned, not overlapped, by (long) divided memory area
 | |
| 	; Args: to,from,length
 | |
| 	;
 | |
| 
 | |
| 	begcode	bmove_align
 | |
| 	public	_bmove_align
 | |
| _bmove_align	proc near
 | |
| 	fix_es	1
 | |
| 	mov	edx,edi
 | |
| 	mov	eax,esi
 | |
| 	mov	edi,P-SIZEPTR[esp]	;to
 | |
| 	mov	esi,P[esp]		;from
 | |
| 	mov	ecx,P+SIZEPTR[esp]	;length
 | |
| 	add	cx,3			;fix if not divisible with long
 | |
| 	shr	cx,2
 | |
| 	rep	movsd
 | |
| 	mov	esi,eax
 | |
| 	mov	edi,edx
 | |
| 	ret
 | |
| _bmove_align	ENDP
 | |
| 	endcode bmove_align
 | |
| 
 | |
| 	;
 | |
| 	; Move a string from higher to lower
 | |
| 	; Arg from+1,to+1,length
 | |
| 	;
 | |
| 
 | |
| 	begcode	bmove_upp
 | |
| 	public	_bmove_upp
 | |
| _bmove_upp	proc near
 | |
| 	fix_es
 | |
| 	std				; Work downward
 | |
| 	mov	edx,edi
 | |
| 	mov	eax,esi
 | |
| 	mov	edi,P-SIZEPTR[esp]	;p1
 | |
| 	mov	esi,P[esp]		;p2
 | |
| 	mov	ecx,P+SIZEPTR[esp]
 | |
| 	dec	edi			; Don't move last arg
 | |
| 	dec	esi
 | |
| 	rep	movsb			; One byte a time because overlap !
 | |
| 	cld				; C compilator wants cld
 | |
| 	mov	esi,eax
 | |
| 	mov	edi,edx
 | |
| 	ret
 | |
| _bmove_upp ENDP
 | |
| 	endcode bmove_upp
 | |
| 
 | |
| 	;
 | |
| 	; Append fillchars to string
 | |
| 	; Args: dest,len,fill
 | |
| 	;
 | |
| 
 | |
| 	begcode	strappend
 | |
| 	public	_strappend
 | |
| _strappend	proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es  1
 | |
| 	push	edi
 | |
| 	mov	edi,P[ebp]		; Memory pointer
 | |
| 	mov	ecx,P+SIZEPTR[ebp]	; Length
 | |
| 	clr	eax			; Find end of string
 | |
| 	repne	scasb
 | |
| 	jnz	sa_99			; String to long, shorten it
 | |
| 	movzx	eax,byte ptr P+(2*SIZEPTR)[ebp]	; Fillchar
 | |
| 	dec	edi			; Point at end null
 | |
| 	inc	ecx			; rep made one dec for null-char
 | |
| 	q_stos				; Store al in string
 | |
| sa_99:	mov	BYTE PTR [edi],0	; End of string
 | |
| 	pop	edi
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strappend	ENDP
 | |
| 	endcode strappend
 | |
| 
 | |
| 	;
 | |
| 	; Find if string contains any char in another string
 | |
| 	; Arg: str,set
 | |
| 	; Ret: Pointer to first found char in str
 | |
| 	;
 | |
| 
 | |
| 	begcode strcont
 | |
| 	PUBLIC	_strcont
 | |
| _strcont proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es	1
 | |
| 	mov	edx,edi
 | |
| 	push	esi
 | |
| 	mov	esi,P[ebp]		; str
 | |
| 	mov	ecx,P+SIZEPTR[ebp]	; Set
 | |
| 	clr	ah			; For endtest
 | |
| 	jmps	sc_60
 | |
| 
 | |
| sc_10:	scasb
 | |
| 	jz	sc_fo			; Found char
 | |
| sc_20:	cmp	ah,[edi]		; Test if null
 | |
| 	jnz	sc_10			; Not end of set yet
 | |
| 	inc	esi			; Next char in str
 | |
| sc_60:	mov	edi,ecx			; edi = Set
 | |
| 	mov	al,[esi]		; Test if this char exist
 | |
| 	and	al,al
 | |
| 	jnz	sc_20			; Not end of string
 | |
| 	clr	esi			; Return Null
 | |
| sc_fo:	mov	eax,esi			; Char found here
 | |
| 	mov	edi,edx			; Restore
 | |
| 	pop	esi
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strcont	ENDP
 | |
| 	endcode strcont
 | |
| 
 | |
| 	;
 | |
| 	; Found end of string
 | |
| 	; Arg: str
 | |
| 	; ret: Pointer to end null
 | |
| 	;
 | |
| 
 | |
| 	begcode strend
 | |
| 	public	_strend
 | |
| _strend	proc near
 | |
| 	fix_es	1
 | |
| 	mov	edx,edi			; Save
 | |
| 	mov	edi,P-SIZEPTR[esp]	; str
 | |
| 	clr	eax			; Find end of string
 | |
| 	mov	ecx,eax
 | |
| 	dec	ecx			; ECX = -1
 | |
| 	repne	scasb
 | |
| 	mov	eax,edi
 | |
| 	dec	eax
 | |
| 	mov	edi,edx			; Restore
 | |
| 	ret
 | |
| _strend	endp
 | |
| 	endcode strend
 | |
| 
 | |
| 	;
 | |
| 	; Make a string with len fill-chars and endnull
 | |
| 	; Args: dest,len,fill
 | |
| 	; Ret:  dest+len
 | |
| 	;
 | |
| 
 | |
| 	begcode	strfill
 | |
| 	public	_strfill
 | |
| _strfill proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es  1
 | |
| 	push	edi
 | |
| 	mov	edi,P[ebp]		; Memory pointer
 | |
| 	mov	ecx,P+SIZEPTR[ebp]	; Length
 | |
| 	movzx	eax,byte ptr P+(2*SIZEPTR)[ebp]	; Fill
 | |
| 	q_stos
 | |
| 	mov	BYTE PTR [edi],0	; End NULL
 | |
| 	mov	eax,edi			; End i DX:AX
 | |
| 	pop	edi
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strfill endp
 | |
| 	endcode strfill
 | |
| 
 | |
| 	;
 | |
| 	; Find a char in or end of a string
 | |
| 	; Arg: str,char
 | |
| 	; Ret: pointer to found char or NullS
 | |
| 	;
 | |
| 
 | |
| 	begcode strcend
 | |
| 	public	_strcend
 | |
| _strcend proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es  1
 | |
| 	mov	edx,edi
 | |
| 	mov	edi,P[ebp]		; str
 | |
| 	mov	ah,P+SIZEPTR[ebp]	; search
 | |
| 	clr	al			; for scasb to find end
 | |
| 
 | |
| @@:	cmp	ah,[edi]
 | |
| 	jz	@F			; Found char
 | |
| 	scasb
 | |
| 	jnz	@B			; Not end
 | |
| 	dec 	edi			; Not found, point at end of string
 | |
| @@:	mov	eax,edi
 | |
| 	mov	edi,edx			; Restore
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strcend	ENDP
 | |
| 	endcode strcend
 | |
| 
 | |
| 	;
 | |
| 	; Test if string has a given suffix
 | |
| 	;
 | |
| 
 | |
| 	begcode is_prefix
 | |
| 	public	_is_prefix
 | |
| _is_prefix proc near
 | |
| 	fix_es	1
 | |
| 	mov	edx,edi			; Save edi
 | |
| 	mov	eax,esi			; Save esi
 | |
| 	mov	esi,P[esp]		; get suffix
 | |
| 	mov	edi,P-SIZEPTR[esp]	; s1
 | |
| 	push	eax			; push esi
 | |
| 	mov	eax,1			; Ok and zero-test
 | |
| @@:	cmp	ah,[esi]
 | |
| 	jz	suf_ok			; End of string; found suffix
 | |
| 	cmpsb				; Compare strings
 | |
| 	jz	@B			; Same, possible prefix
 | |
| 	xor	eax,eax			; Not suffix
 | |
| suf_ok:	pop	esi
 | |
| 	mov	edi,edx
 | |
| 	ret
 | |
| _is_prefix endp
 | |
| 	endcode	_is_prefix
 | |
| 
 | |
| 	;
 | |
| 	; Find a substring in string
 | |
| 	; Arg: str,search
 | |
| 	;
 | |
| 
 | |
| 	begcode strstr
 | |
| 	public	_strstr
 | |
| _strstr proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es	1
 | |
| 	push	EDI
 | |
| 	push	ESI
 | |
| 	mov	esi,P[ebp]		; str
 | |
| 	mov	edi,P+SIZEPTR[ebp]	; search
 | |
| 	mov	ecx,edi
 | |
| 	inc	ecx			; ECX = search+1
 | |
| 	mov	ah,[edi]		; AH = First char in search
 | |
| 	jmps	sf_10
 | |
| 
 | |
| sf_00:	mov	esi,edx			; si = Current str-pos
 | |
| sf_10:	mov	al,[esi]		; Test if this char exist
 | |
| 	and	al,al
 | |
| 	jz	sf_90			; End of string, didn't find search
 | |
| 	inc	esi
 | |
| 	cmp	al,ah
 | |
| 	jnz	sf_10			; Didn't find first char, continue
 | |
| 	mov	edx,esi			; Save str-pos in EDX
 | |
| 	mov	edi,ecx
 | |
| sf_20:	cmp	BYTE PTR [edi],0
 | |
| 	jz	sf_fo			; Found substring
 | |
| 	cmpsb
 | |
| 	jz	sf_20			; Char ok
 | |
| 	jmps	sf_00			; Next str-pos
 | |
| 
 | |
| sf_90:	mov	edx,1			; Return Null
 | |
| sf_fo:	mov	eax,edx			; Char found here
 | |
| 	dec	eax			; Pointed one after
 | |
| 	pop	ESI
 | |
| 	pop	EDI
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strstr endp
 | |
| 	endcode strstr
 | |
| 
 | |
| 	;
 | |
| 	; Find a substring in string, return index
 | |
| 	; Arg: str,search
 | |
| 	;
 | |
| 
 | |
| 	begcode	strinstr
 | |
| 	public	_strinstr
 | |
| _strinstr proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	push	P+SIZEPTR[ebp]		; search
 | |
| 	push	P[ebp]			; str
 | |
| 	call	_strstr
 | |
| 	add	esp,SIZEPTR*2
 | |
| 	or	eax,eax
 | |
| 	jz	si_99			; Not found, return NULL
 | |
| 	sub	eax,P[ebp]		; Pos from start
 | |
| 	inc	eax			; And first pos = 1
 | |
| si_99:	pop	ebp
 | |
| 	ret
 | |
| _strinstr	endp
 | |
| 	endcode strinstr
 | |
| 
 | |
| 	;
 | |
| 	; Make a string of len length from another string
 | |
| 	; Arg: dst,src,length
 | |
| 	; ret: end of dst
 | |
| 	;
 | |
| 
 | |
| 	begcode strmake
 | |
| 	public	_strmake
 | |
| _strmake proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es	1
 | |
| 	push	EDI
 | |
| 	push	ESI
 | |
| 	mov	edi,P[ebp]		; dst
 | |
| 	mov	esi,P+SIZEPTR[ebp]	; src
 | |
| 	mov	ecx,P+SIZEPTR*2[ebp]	; Length of memory-area
 | |
| 	clr	al			; For test of end-null
 | |
| 	jcxz	sm_90			; Nothing to move, put zero at end.
 | |
| 
 | |
| @@:	cmp	al,[esi]		; Next char to move
 | |
| 	movsb				; move arg
 | |
| 	jz	sm_99			; last char, we are ready
 | |
| 	loop	@B			; Continue moving
 | |
| sm_90:	mov	BYTE PTR [edi],al	; Set end pos
 | |
| 	inc	edi			; Fix that di points at end null
 | |
| sm_99:	dec	edi			; di points now at end null
 | |
| 	mov	eax,edi			; Ret value in DX:AX
 | |
| 	pop	ESI
 | |
| 	pop	EDI
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strmake	ENDP
 | |
| 	endcode strmake
 | |
| 
 | |
| 	;
 | |
| 	; Find length of string with maxlength
 | |
| 	; arg: str,maxlength
 | |
| 	; ret: length
 | |
| 	;
 | |
| 
 | |
| 	begcode	strnlen
 | |
| 	public	_strnlen
 | |
| _strnlen proc near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es	1
 | |
| 	push	edi
 | |
| 	mov	edi,P[ebp]		; Str
 | |
| 	mov	ecx,P+SIZEPTR[ebp]	; length
 | |
| 	mov	edx,edi			; Save str to calc length
 | |
| 	jcxz	sn_10			; Length = 0
 | |
| 	clr	al			; Find end of string
 | |
| 	repne	scasb			; Find strend or length
 | |
| 	jnz	sn_10
 | |
| 	dec	edi			; DI points at last null
 | |
| sn_10:	mov	eax,edi
 | |
| 	sub	eax,edx			; Ax = length
 | |
| 	pop	edi
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strnlen	ENDP
 | |
| 	endcode strnlen
 | |
| 
 | |
| 	;
 | |
| 	; Move a string with max len chars
 | |
| 	; arg: dst,src,len
 | |
| 	; ret: pos to first null or dst+len
 | |
| 
 | |
| 	begcode	strnmov
 | |
| 	public	_strnmov
 | |
| _strnmov PROC near
 | |
| 	push	ebp
 | |
| 	mov	ebp,esp
 | |
| 	fix_es	1
 | |
| 	push	EDI
 | |
| 	push	ESI
 | |
| 	mov	edi,P[ebp]		; dst
 | |
| 	mov	esi,P+SIZEPTR[ebp]	; src
 | |
| 	mov	ecx,P+(SIZEPTR*2)[ebp]	; length
 | |
| 	jcxz	snm_99			; Nothing to do
 | |
| 	clr	al			; For test of end-null
 | |
| 
 | |
| @@:	cmp	al,[esi]		; Next char to move
 | |
| 	movsb				; move arg
 | |
| 	jz	snm_20			; last char, fill with null
 | |
| 	loop	@B			; Continue moving
 | |
| 	inc	edi			; Point two after last
 | |
| snm_20:	dec	edi			; Point at first null (or last+1)
 | |
| snm_99:	mov	eax,edi			; Pointer at last char
 | |
| 	pop	ESI
 | |
| 	pop	EDI
 | |
| 	pop	ebp
 | |
| 	ret
 | |
| _strnmov	ENDP
 | |
| 	endcode strnmov
 | |
| 
 | |
| ;
 | |
| ; Zortech has this one in standard library
 | |
| ;
 | |
| 
 | |
| 	begcode strmov
 | |
| 	public	_strmov
 | |
| _strmov proc	near
 | |
| 	mov	ecx,esi			; Save old esi and edi
 | |
| 	mov	edx,edi
 | |
| 	mov	esi,P[esp]		; get source pointer (s2)
 | |
| 	mov	edi,P-SIZEPTR[esp]	; EDI -> s1
 | |
| 	fix_es	1
 | |
| @@:	mov	al,[esi]
 | |
| 	movsb				; move arg
 | |
| 	and	al,al
 | |
| 	jnz	@B			; Not last
 | |
| 	mov	eax,edi
 | |
| 	dec	eax
 | |
| 	mov	esi,ecx			; Restore args
 | |
| 	mov	edi,edx
 | |
| 	ret
 | |
| _strmov endp
 | |
| 	endcode strmov
 | |
| 
 | |
| endif ; M_I386
 | |
| 
 | |
| 	END
 |