|  
โพรซีเดอร์ และมาโคร ในแอสเซมบลี้
 |  | 
| ปรับปรุง : 2562-02-01 (ปรับ template) | 
| Digital logic | OS | คำสั่งดอส | Batch | Debug | Assembly | GWBasic | Docker | | 
| 1. การเขียน procedure พิมพ์ A - เขียน procedure นอก .code หรือ segment เหมือน macro ไม่ได้ - การเรียกใช้ procedure ต้องขึ้นต้นด้วยคำว่า call - ก่อน endp ต้องสั่ง ret เพื่อกลับมายังบรรทัดที่ call       .model  small
      .data           
      .code
x:    mov ax,@data   
      mov ds,ax
      call oho1    
y:    mov ah,4ch     
      int 21h
; === procedure ===
oho1 proc near      
      mov ah,2h     
      mov dl,41h      
      int 21h      
      ret
oho1 endp
      .stack  200h    
      end x
 | 
| 2. การเขียนมากกว่า 1 procedure - procedure ถูกเรียกใช้ได้หลายครั้ง เช่นเดียวกับ macro - สามารถประการ procedure ใต้ .code แล้วเรียกใช้ภายหลังได้       .model  small
      .data           
      .code
x:    mov ax,@data   
      mov ds,ax
      call oho1   
      call oho2
      call oho1   
      call oho2      
y:    mov ah,4ch     
      int 21h
; === procedure ===
oho1 proc near      
      mov ah,2h     
      mov dl,41h      
      int 21h      
      ret
oho1 endp
oho2 proc near      
      mov ah,2h     
      mov dl,61h      
      int 21h      
      ret
oho2 endp
      .stack  200h    
      end x
 | 
| 3. การเขียน macro พิมพ์ A - ประกาศ macro ก่อนเข้าโปรแกรม จะประกาศเหมือน procedure ไม่ได้ - เพียงพิมพ์ชื่อมาโคร ก็จะหมายถึงเรียกใช้ macro helloa  macro
        mov     ah,2
        mov     dl,41h
        int     21h
        endm
; === end of macro ===
        .model  small
        .data
        .code
pmain   proc    far      
        push    ds      ; 1 of 5 line required for .exe
        mov     ax,0    ; 2 clear ax by xor  ax,ax
        push    ax      ; 3 send ax to stack
        mov     ax,@data
        mov     ds,ax
        helloa
        ret
pmain   endp
        .stack  200h    ; not required (if have, will not warning in link)
        end     pmain
 | 
| 4. การเขียนมากกว่า 1 macro setproc macro
        push    ds      ; 1 of 5 line required for .exe
        mov     ax,0    ; 2 clear ax by xor  ax,ax
        push    ax      ; 3 send ax to stack
        mov     ax,@data
        mov     ds,ax
        endm	
prtout  macro
        mov     ah,9
        lea     dx,msg
        int     21h
        endm
; =============
; Main program
; =============
        .model  small
        .data
msg     db      'This is the program testing $'
        .code
pmain   proc    far     ; can not change far to near 
        setproc
        prtout
        ret
pmain   endp
        .stack  200h    ; not required
        end     pmain
 | 
| 5. การเขียน macro เพิ่มค่าให้กับ al ทีละ 1 แล้วพิมพ์ โดยเรียก macro 5 ครั้ง love    macro
        mov     bl,01h
        add     al,bl
        mov     sum,al
        mov     dl,sum
        mov     ah,2
        int     21h
        endm
; =============
; Main program
; =============
        .model  small
        .data
n       db      30h
sum     db  ?
        .code
pmain   proc    far     ; can not change far to near 
        push    ds      ; 1 of 5 line required for .exe
        mov     ax,0    ; 2 clear ax by xor  ax,ax
        push    ax      ; 3 send ax to stack
        mov     ax,@data
        mov     ds,ax
        mov     al,n
        love
        love
        love
        love
        love
        love
pmain   endp
        end     pmain
 | 
| 6. โปรแกรมพิมพ์ 1 ถึง 15 โดยใช้ 2 macro - macro ชื่อ jack ใช้พิมพ์ 1 ถึง 9 โดยแยกตัวเลขละ 1 บรรทัด - macro ชื่อ jack ใช้พิมพ์ 10 ถึง 15 โดยใช้การแสดง 1 และ 0 ถึง 5 แยกกันคนละตัวอักษร - ผู้เรียนต้องเข้าใจว่า 30h คือเลข 0 
jack  macro
      mov    al,cl
      mov    bl,01h
      add    al,bl
      mov    sum,al
      mov    dl,sum
      mov    ah,2
      int    21h
      mov    cl,al
      lea    dx,lf
      mov    ah,9
      int    21h
      endm
mary  macro
      mov    dl,31h
      mov    ah,2
      int    21h
      mov    dl,bl
      mov    ah,2
      int    21
      mov    al,cl
      mov    bl,01h
      add    al,bl
      mov    sum,al
      mov    dl,sum
      mov    ah,2
      int    21h
      mov    cl,al
      lea    dx,lf
      mov    ah,9
      int    21h
      endm
; =============
; Main program
; =============
      .model small
      .data
n1    db     30h
n2    db     2fh
last1 db     39h
last2 db     35h
lf    db     0dh,0ah,'$'
sum   db     ?
      .code
pmain proc   far     ; can not change far to near 
      push   ds      ; 1 of 5 line required for .exe 
      mov    ax,0    ; 2 clear ax by xor  ax,ax
      push   ax      ; 3 send ax to stack
      mov    ax,@data
      mov    ds,ax
      mov    cl,n1
boy:  jack
      cmp    cl,last1
      jl     boy
      mov    cl,n2
girl: mary
      cmp    cl,last2
      jl     girl
      ret
pmain endp
      .stack 10h
      end    pmain
 | 
| 7. รับตัวอักษร และเลือกด้วย CMP ไปทำ Macro - ถ้ารับตัวอักษร a ก็จะแสดงตัวอักษร C - ถ้ารับตัวอักษรที่ไม่ใช่ a ก็จะไม่แสดงอะไรเลย burin1	macro
	mov     ah,2
        mov     dl,43h ; C
        int     21h
	endm
	.model  small
        .data
        .code
pmain   proc    far      
        push    ds     
        mov     ax,0   
        push    ax     
        mov     ax,@data
        mov     ds,ax
        mov     ah,8
        int     21h
        cmp     al,61h ; a
	je      work1
        jmp     bye
work1:  burin1
bye:    ret
pmain   endp
        .stack  200h   
        end     pmain
 | 
| 8. แปลงโปรแกรมเข้ารหัส และถอดรหัสส่วนหนึ่งของ Kailas Jagtap เป็น macro - โปรแกรมนี้นำมาจาก http://www.thaiall.com/assembly/loopbasic.htm - มีต้นฉบับที่ http://www.laynetworks.com/Assembly%20Program7.htm โดย Kailas Jagtap - โปรแกรมนี้แบ่งเป็น 4 procedure และ 4 macro - ลบ Comment ออกเพื่อให้ดูสั้นลง - ปรับไปอยู่ใน word ที่ kailas.docx - http://www.thaiall.com/assembly/interrupt.htm - http://www.emu8086.com/assembler_tutorial/8086_bios_and_dos_interrupts.html ; ตัวอย่างการย้ายบางส่วนของโปรแกรมมาไว้ใน macro ; โปรแกรมนี้ ได้แยกการทำงาน readnext ที่ทำงานเหมือนกันแต่เขียนต่างกันไว้ คือ encrypt เขียน readnext ใน label ; แต่ decrypt จะเขียน readnext ออกมาเป็นอีก macro หนึ่ง เป็นการทำงานร่วมกันของ burin3 และ burin4 ; เขียนบล็อก http://www.thaiall.com/blog/burin/3691/ burin4 macro ; move readnext1 in macro mov ah,3fh ; อ่านข้อมูลที่ handle ไว้ เข้ามาในหน่วยความจำ 512 Bytes mov bx,handle mov cx,512 lea dx,buffer int 21h jc err_out1 ; ถ้าอ่านได้ จะทำให้ carry เป็น 0 mov save_ax,ax ; ค่าของ ax คือจำนวนไบท์ที่อ่านได้ cmp ax,0 je done4 ; ถ้ากระโดดไป แสดงว่าจบแฟ้ม เพราะขนาดของ ax = 0 call decorrupt mov ah,40h ; เขียนข้อมูลขนาดเท่า save_ax ลงไปใน newhandle mov bx,newhandle mov cx,save_ax lea dx,buffer ; ข้อมูลที่ผ่านการยกเลิกแผลงหรือไขว้ (decorrupt) มาแล้ว int 21h jc err_out1 ; ถ้าเขียนได้ จะทำให้ carry เป็น 0 jmp readnext1 ; กระโดดไปเริ่มอ่านข้อมูลต่อจากจุดเดิม endm burin3 macro ; move decr in macro call disp_nl mov ah,9 ; แสดงข้อความว่า รับชื่อแฟ้มที่ต้องการ DECRYPT lea dx,s3 int 21h lea dx,fname ; กำหนดชื่อแฟ้มอ้างอิง เมื่อรับผ่านแป้นพิมพ์ call getname mov ah,3dh ; เปิดแฟ้มข้อมูล lea dx,fname+2 mov al,0 int 21h mov handle,ax ; เก็บตำแหน่งอ้างอิงแฟ้ม jc err_out ; ถ้าเปิดได้ จะทำให้ carry เป็น 0 mov ah,9 ; แสดงข้อความว่า รับชื่อแฟ้มที่ต้องการ DECRYPT lea dx,s33 int 21h lea dx,newf1 ; กำหนดชื่อแฟ้มอ้างอิง เมื่อรับผ่านแป้นพิมพ์ call getname mov ah,3ch ; สร้างไฟล์ใหม่ lea dx,newf1+2 mov cx,0 int 21h mov newhandle,ax ; เก็บตำแหน่งอ้างอิงแฟ้ม jc err_out ; ถ้าสร้างได้ จะทำให้ carry เป็น 0 endm burin2 macro ; move encr in macro call disp_nl mov ah,9 ; แสดงข้อความว่า รับชื่อแฟ้มที่ต้องการ ENCRYPT lea dx,s2 int 21h lea dx,fname ; กำหนดชื่อแฟ้มอ้างอิง เมื่อรับผ่านแป้นพิมพ์ call getname mov ah,3dh ; เปิดแฟ้มข้อมูล lea dx,fname+2 mov al,0 int 21h mov handle,ax ; เก็บตำแหน่งอ้างอิงแฟ้ม jc err_out2 ; ถ้าเปิดได้ จะทำให้ carry เป็น 0 mov ah,9 ; แสดงข้อความว่า รับชื่อแฟ้มที่ต้องการ ENCRYPT lea dx,s22 int 21h lea dx,newf1 ; กำหนดชื่อแฟ้มอ้างอิง เมื่อรับผ่านแป้นพิมพ์ call getname mov ah,3ch ; สร้างไฟล์ใหม่ lea dx,newf1+2 mov cx,0 int 21h mov newhandle,ax ; เก็บตำแหน่งอ้างอิงแฟ้ม jc err_out ; ถ้าสร้างได้ จะทำให้ carry เป็น 0 readnext: mov ah,3fh ; อ่านข้อมูลที่ handle ไว้ เข้ามาในหน่วยความจำ 512 Bytes mov bx,handle mov cx,512 lea dx,buffer int 21h jc err_out ; ถ้าอ่านได้ จะทำให้ carry เป็น 0 mov save_ax,ax ; ค่าของ ax คือจำนวนไบท์ที่อ่านได้ cmp ax,0 je done1 ; ถ้ากระโดดไป แสดงว่าจบแฟ้ม เพราะขนาดของ ax = 0 call corrupt mov ah,40h ; เขียนข้อมูลขนาดเท่า save_ax ลงไปใน newhandle mov bx,newhandle mov cx,save_ax lea dx,buffer ; ข้อมูลที่ผ่านการ แผลงหรือไขว้ (corrupt) มาแล้ว int 21h jc err_out ; ถ้าเขียนได้ จะทำให้ carry เป็น 0 jmp readnext ; กระโดดไปเริ่มอ่านข้อมูลต่อจากจุดเดิม err_out2: jmp err_out endm burin1 macro ; move rpt_disp in macro mov ah,9 ; แสดงคำว่า E หรือ D lea dx,s1 int 21h mov ah,1 ; รับตัวอักษร 1 ตัว int 21h cmp al,'E' je encr ; กระโดดไปยัง encr (ENCRYPT) cmp al,'e' je encr cmp al,'D' je decr2 ; กระโดดไปยัง decr2 (DECRYPT) cmp al,'d' je decr2 cmp al,'X' je stop2 cmp al,'x' je stop2 mov ah,9 ; แสดงข้อความว่าตัวเลือกที่ส่งมา ไม่ถูกต้อง lea dx,s4 int 21h jmp rpt_disp endm .model small .data s0 db 0dh,0ah,'PROGRAM FOR FILE ENCRYPTION AND DECRYPTION ..By Kailas Jagtap$' s1 db 0dh,0ah,'Do you want to ENCRYPT(E) or DECRYPT(D) a file ? : $' s2 db 0dh,0ah,'Enter Name of File(to be encrypted): $' s22 db 0dh,0ah,'Enter Name of File(which will store encrypted data): $' s3 db 0dh,0ah,'Enter Name of encrypted File(to be decrypted): $' s33 db 0dh,0ah,'Enter Name of File(which will store decrypted data): $' s4 db 0dh,0ah,'***************** ERROR!!! - INVALID INPUT ****************** $' s44 db 0dh,0ah,'***************** ERROR IN FILE OPERATION !!!!!! ************ $' s5 db 0dh,0ah,'$' s6 db 0dh,0ah,'.............. File ENCRYPTED Successfully ! ................. $' s7 db 0dh,0ah,'.............. File DECRYPTED Successfully ! ................. $' fname db 80 db 0 db 80 dup(0) newf1 db 80 db 0 db 80 dup(0) save_ax dw 1 dup(0) buffer db 512 dup(0) endbuffer db '$' handle dw ? newhandle dw ? .code start: mov ax,@data mov ds,ax call disp_nl mov ah,9 lea dx,s0 int 21h rpt_disp: burin1 decr2: jmp decr1 stop2: jmp stop1 encr: burin2 decr1: jmp decr stop1: jmp stop done1: lea dx,s6 ; พิมพ์ว่า การเข้ารหัส สำเร็จลุล่วงไปด้วยดี jmp done3 done2: lea dx,s7 ; พิมพ์ว่า การถอดรหัส สำเร็จลุล่วงไปด้วยดี done3: mov ah,9 int 21h jmp done stop: mov ah,4ch int 21h getname proc near mov ah,0ah ; รับข้อมูล String จากแป้มพิมพ์ int 21h mov si,dx ; บันทึกตำแหน่งของชื่อแฟ้มที่รับจากแป้นพิมพ์ mov bl,[si+1] add si,02 sub bh,bh mov BYTE PTR[si+bx],0 ret getname endp err_out: mov ah,9 lea dx,s44 ; การดำเนินการกับแฟ้ม ผิดพลาด int 21h jmp stop done: mov ah,3eh ; ปิดแฟ้มให้เรียบร้อย ก่อนจบการทำงาน mov bx,handle int 21h mov bx,newhandle int 21h jmp stop done4: jmp done2 corrupt proc near lea si,buffer ; ส่งตำแหน่ง buffer ให้ si (Load effective adress) mov cx,save_ax ; ค่า cx คู่กับ loop บอกว่า จะให้ทำใน loop กี่รอบ again: mov al,[si] ; สำเนาค่าของ si ที่ตำแหน่งปัจจุบัน ให้กับ al add al,07h ; บวกอีก 7 ให้กับ al เช่นเดิมคือ a ก็จะเป็น h mov [si],al ; ส่งผลการ corrupt ไปแทนค่าในตำแหน่งเดิม inc si ; ตรวจ loop again ret corrupt endp err_out1: jmp err_out decr: burin3 readnext1: burin4 decorrupt proc near lea si,buffer ; ส่งตำแหน่ง buffer ให้ si (Load effective adress) mov cx,512 ; ค่า cx คู่กับ loop บอกว่า จะให้ทำใน loop กี่รอบ again1: mov al,[si] ; สำเนาค่าของ si ที่ตำแหน่งปัจจุบัน ให้กับ al sub al,07h ; ลบอีก 7 ให้กับ al เช่นเดิมคือ 8 ก็จะเป็น 1 mov [si],al ; ส่งผลการ corrupt ไปแทนค่าในตำแหน่งเดิม inc si loop again1 ret decorrupt endp disp_nl proc near mov ah,9 lea dx,s5 int 21h ret disp_nl endp .stack 200h end start | 
| 
 | 
| "Imagination is more important than knowledge" - Albert Einstein |