nguyenanhque .asm4
title:LẬP TRÌNH VỚI HỢP NGỮ
date:28-10-2008
sender:Người Đẹp
source:
type:ppt
LẬP TRÌNH VỚI HỢP NGỮ
Gv: Lê Minh Triết
Quy trình tạo và chạy chương trình
Bộ hợp dịch ASM có hai trình cơ bản là
TASM.EXE (trình hợp dịch)
TLINK.EXE (trình liên kết)
Ngoài ra ta còn cần một chương trình dùng để soạn thảo để tạo chương trình nguồn.
! Ta có thể dùng bộ chương trình BorlandC để soạn thảo chương trình nguồn.
Phần 1:
Các bước tiến hành lập trình
Các bước cài đặt và tạo đường dẫn File biên dịch
Chạy file Install.exe trong thư mục BorlandC (BC)
Nhấn nút Enter để bắt đầu cài đặt
Các bước cài đặt và tạo đường dẫn File biên dịch
3. Chọn lại ổ đĩa chứa các tập tin cài đặt
Các bước cài đặt và tạo đường dẫn File biên dịch
Kiểm tra đường dẫn chứa các tập tin cài đặt
Các bước cài đặt và tạo đường dẫn File biên dịch
4. Sửa lại thư mục cài đặt (nhấn Enter)
Các bước cài đặt và tạo đường dẫn File biên dịch
E:BorlandC sửa lại là C:BC hay C:BorlandC
Các bước cài đặt và tạo đường dẫn File biên dịch
Sau khi sửa xong, nhấn ESC để thoát khỏi màn hình chỉnh sửa đường dẫn.
Chọn Start Installation để bắt đầu cài đặt.
Các bước cài đặt và tạo đường dẫn File biên dịch
Khi cài đặt có 1 vài file bị lỗi, ta nhấn nút C để tiếp tục
Các bước cài đặt và tạo đường dẫn File biên dịch
Nhấn 1 nút bất kỳ để tiếp tục
- ESC để tắt màn hình.
Chú ý tạo đường dẫn (PATH)
Tạo file đường dẫn Path
Tạo 01 file *.bat trong ổ đĩa C:
(hay ổ đĩa đã cài đặt BorlandC)
Các bước tiến hành lập trình
Chạy cửa sổ Run
Gõ lệnh CMD
CD
Chạy File P.bat
Chạy chương trình soạn thảo BC.EXE
Soạn nội dung chương trình nguồn
Lưu lại File với đuôi *.asm
Thoát khỏi BorlandC
Chạy chương trình hợp dịch TASM.EXE
Chạy trình liên kết TLINK.EXE
Thực thi chương trình.
Ví dụ: Soạn chương trình Hello
Soạn tập tin Hello.asm
Khai báo dữ liệu bắt đầu bằng .Data
Khai báo đoạn mã chương trình bắt đầu bằng .Code
Ví dụ: Dịch chương trình Hello
Gõ lệnh Tasm Hello.asm
Thông báo số lỗi
Ví dụ: Dịch chương trình Hello
Gõ lệnh Tasm Hello.asm
Gõ tiếp lệnh Tlink Hello.obj
Chương trình sẽ tạo ra file Hello.exe
Thông báo số lỗi
TỔ CHỨC THANH GHI
Thanh ghi (register) là nơi lưu dữ liệu bên trong CPU
Tùy theo độ dài 8 hay 16 bit và tùy theo chức năng khi đó thanh ghi được dùng để chứa dữ liệu sẽ thao tác hoặc kết quả các phép tính hoặc các địa chỉ dùng để định vị ô nhớ khi cần thiết.
Có tất cả 14 thanh ghi, mỗi thanh ghi dài 16 bit chia thành năm nhóm
Phần 2:
Nhóm thanh ghi đoạn
Gồm 4 thanh ghi: đoạn mã CS, đoạn dữ liệu DS, đoạn bổ sung ES và đoạn stack SS. Là những thanh ghi chứa địa chỉ segment của các ô nhớ khi cần truy xuất.
Thanh ghi đoạn mã CS (Code Segment): Lưu địa chỉ segment chứa chương trình ngôn ngữ máy.
Thanh ghi đoạn dữ liệu DS (Data Segment): Lưu địa chỉ segment của đoạn chứa dữ liệu trong chương trình.
Thanh ghi đoạn bổ sung ES (Extra Segment): Lưu địa chỉ segment của đoạn dữ liệu bổ sung.
Thanh ghi đoạn Stack SS (Stack Segment): Lưu địa chỉ segment của đoạn stack.
4 thanh ghi này có thể truy xuất dữ liệu trên 4 đoạn khác nhau và 1 chương trình chỉ có thể sử dụng cùng một lúc tối đa 4 đoạn.
CPU 80386 có 2 thanh ghi tương tự như ES là FS và GS.
2. Nhóm thanh ghi đa dụng
(general register)
Gồm bốn thanh ghi AX, BX, CX, DX. Các thanh ghi này có thể xem như một thanh ghi 16 bit hoặc hai thanh ghi mỗi thanh ghi 4 bit:
AX = AH + AL
BX = BH + BL
CX = CH + CL
DX = DH + DL
CPU 80386 có thể kéo dài đến 32 bit tạo thành thanh ghi EAX, EBX, ECX, EDX.
Thanh ghi tích lũy AX (Accumulator register): thường dùng để lưu số nhân, số chia trong các phép toán nhân, chia, các phép tính số học, logic và chuyển dữ liệu.
VD: MUL BH ; AX ← AL*BH
Thanh ghi cơ sở BX (Base register): thường dùng để định vị bộ nhớ.
VD: MOV [BX], AX ; Lấy nội dung thanh ghi AX đưa vào ô nhớ
; có địa chỉ segment là DS và địa chỉ offset BX.
Thanh ghi đếm CX (Count register): dùng để định số lần lặp của vòng lặp.
Thanh ghi dữ liệu DX (Data register): dùng để lưu kết quả củ các phép toán nhân và chia, định địa chỉ cổng trong các lệnh nhập xuất cổng.
VD: MOV AL, 62 ; AL ← 62
MOV DX, 1000 ; DX ← 1000
OUT DX, AL ; Đưa nội dung của AL (tức 62) ra cổng 1000
1. Cú pháp lệnh hợp ngữ
Một chương trình hợp ngữ gồm các Statement (mệnh đề) được viết liên tiếp nhau , mỗi Statement được viết trên 1 dòng. Một Statement có thể là:
1 lệnh (Instruction) : được chuyển thành mã máy.
1 chỉ dẫn (Assembler directive) : không chuyển thành mã máy
Các lệnh gồm 4 trường :
Name Operation Operand(s) Comment
Các trường cách nhau ít nhất 1 khoảng trắng hoặc 1 TAB
Ví dụ: START : MOV CX,5 ; khơỉ tạo thanh ghi CX
Hay một chỉ dẫn của ASM :
Ví dụ: MAIN PROC ; tạo một thủ tục có tên là MAIN
Phần 3:
1.1 Trường Tên
(Name Field)
Trường tên dùng cho nhãn lệnh, tên thủ tục và tên biến. ASM sẽ chuyển tên thành địa chỉ bộ nhớ .
Tên có thể dài từ 1 đến 31 ký tự .
Trong tên chứa các ký tự từ a-z, các số và các ký tự đặc biệt sau: ? ,@ , _ , $ và dấu.
Không được phép có ký tự trống trong phần tên.
Tên không được bắt đầu bằng một số .
ASM không phân biệt giữa ký tự viết thường và viết hoa .
Các ví dụ về tên hợp lệ và không hợp lệ trong ASM.
Tên hợp lệ Tên không hợp lệ
COUNTER1 TWO WORDS
@CHARACTER 2ABC
SUM_OF_DIGITS A45.28
DONE? YOU&ME
.TEST ADD-REPEAT
1.2 Trường toán tử
( operation field)
Đối với 1 lệnh trường toán tử chưá ký hiệu (Symbol) của phép toán (Operation code = OPCODE). ASM sẽ chuyển ký hiệu phép toán thành mã máy .
Thông thường ký hiệu mã phép toán mô tả chức năng của phép toán. Ví dụ: ADD, SUB, INC, DEC, INT...
Đối với chỉ dẫn của ASM, trường toán tử chưá một opcode giả (pseudo operation code = pseudo-op). ASM không chuyển pseudo-op thành mã máy mà hướng dẫn ASM thực hiện một việc gì đó ví dụ tạo ra một thủ tục, định nghĩa các biến ...
1.3 Trường các toán hạng
(Operand(s) field)
Trong 1 lệnh, trường toán hạng chỉ ra các số liệu tham gia trong lệnh đó.
1 lệnh có thể không có toán hạng , có 1 hoặc 2 toán hạng .
Ví dụ: NOP ; không có toán hạng
INC AX ; 1 toán hạng
ADD WORD1,2 ; 2 toán hạng,
;cộng 2 với nội dung của từ nhớ WORD1
Trong các lệnh 2 toán hạng toán hạng đầu là toán hạng đích (destination operand). Toán hạng đích thường là thanh ghi hoặc vị trí nhớ dùng để lưu trữ kết quả. Toán hạng thứ hai là toán hạng nguồn. Toán hạng nguồn thường không bị thay đổi sau khi thực hiện lệnh .
Đối với một chỉ dẫn của ASM, trường toán hạng chứa một hoặc nhiều thông tin mà ASM dùng để thực thi chỉ dẫn .
1.4 Trường chú thích
(Comment field)
Trường chú thích là một tuỳ chọn trong ngôn ngữ ASM. Lập trình viên dùng trường chú thích để thuyết minh về câu lệnh. Điều này là cần thiết vì ngôn ngữ ASM là ngôn ngữ cấp thấp (low level) vì vậy sẽ rất khó hiểu chương trình nếu nó không được chú thích một cách đầy đủ và rõ ràng. Tuy nhiên không nên có chú thích đối với mọi dòng của chương trình, kể cả những lệnh mà ý nghĩa của nó đã rất rõ ràng.
2. Các kiểu số liệu trong
chương trình hợp ngữ
CPU chỉ làm việc với các số nhị phân. Vì vậy ASM phải chuyển tất cả các loại số liệu thành số nhị phân.
Trong một chương trình hợp ngữ cho phép biểu diễn số liệu dưới dạng nhị phân, thập phân hoặc thập lục phân.
2.1 Các số
1 số nhị phân là 1 dãy các bit 0, 1 và kết thúc bằng b hoặc B
1 số thập phân là 1 dãy các chữ số thập phân và kết thúc bởi d hoặc D (có thể không cần)
1 số hex phải bắt đầu bởi 1 chữ số thập phân và phải kết thúc bởi h hoặc H.
Ví dụ các biểu diễn số hợp lệ và không hợp lệ trong ASM :
Số Loại
10111 thập phân
10111b nhị phân
-2183D thập phân
1B4DH hex
1B4D số hex không hợp lệ
FFFFH số hex không hợp lệ
0FFFFH số hex
2.2 Các ký tự
Ký tự và một chuỗi các ký tự phải được đóng giữa hai dấu ngoặc đơn hoặc hai dấu ngoặc kép.
Ví dụ: 'A' và "HELLO".
Các ký tự đều được chuyển thành mã ASCII.
Do đó trong một chương trình ASM xem khai báo 'A' hay 41h (mã ASCII của A)
là giống nhau
3. Các biến (Variables)
Trong ASM biến đóng vai trò như trong ngôn ngữ cấp cao. Mỗi biến có một loại dữ liệu và nó được gán một địa chỉ bộ nhớ sau khi dịch chương trình. Bảng sau đây liệt kê các toán tử giả dùng để định nghĩa các kiểu số liệu.
PSEUDO-OP STANDS FOR
DB define byte
DW define word (doublebyte)
DD define doubeword (2 từ liên tiếp)
DQ define quadword (4 từ liên tiếp )
DT define tenbytes (10 bytes liên tiếp)
3.1. Kiểu byte
Để định nghĩa biến kiểu byte cú pháp như sau:
NAME DB initial_value
Ví dụ : ALPHA DB 4
- Chỉ dẫn này sẽ gán tên ALPHA cho một byte nhớ trong bộ nhớ mà giá trị ban đầu của nó là 4.
- Nếu giá trị của byte là không xác định thì đặt dấu chấm hỏi ( ?) vào giá trị ban đầu.
Ví dụ : BYT DB ?
Đối với biến kiểu byte vùng giá trị nó lưu trữ được là từ -128 đến 127 đối với số có dấu và 0 đến 255 đối với số không dấu .
3.2. Kiểu word
Định nghĩa một biến kiểu Word như sau:
NAME DW initial_value
Ví dụ : WRD DW -2
Có thể dùng dấu ? để thay thế cho biến từ có giá trị không xác định. Vùng giá trị của biến từ là -32768 đến 32767 đối với số có dấu và 0 đến 56535 đối với số không dấu .
3.3 Mảng (arrays)
Mảng là một loạt các byte nhớ hoặc word nhớ liên tiếp nhau. Ví dụ để định nghĩa 1 mảng 3 byte là B_ARRAY,
giá trị ban đầu là 10h, 20h và 30h ta có thể viết :
B_ARRAY DB 10h,20h,30h
B_ARRAY là tên được gán cho byte đầu tiên
B_ARRAY+1 là tên của byte thứ hai
B_ARRAY+2 là tên của byte thứ ba
Nếu ASM gán địa chỉ offset là 0200h cho mảng B_ARRAY thì nội dung bộ nhớ sẽ như sau :
SYMBOL ADDRESS CONTENTS
B_ARRAY 200h 10h
B_ARRAY+1 201h 20h
B_ARRAY+2 202h 30h
Chỉ dẫn sau đây sẽ định nghĩa một mảng 4 phần tử có tên là W_ARRAY:
W_ARRAY DW 1000,40,29887,329
Giả sử mảng bắt đầu tại 0300h, bộ nhớ như sau:
SYMBOL ADDRESS CONTENTS
W_ARRAY 300h 1000d
W_ARRAY+2 302h 40d
W_ARRAY+4 304h 29887d
W_ARRAY+6 306h 329d
Byte thấp và byte cao của một word
Đôi khi chúng ta cần truy xuất tới byte thấp và byte cao của một biến Word.
Giả sử chúng ta định nghĩa :
WORD1 DW 1234h
Byte thấp của WORD1 chứa 34h
Còn byte cao của WORD1 chứa 12h
Ký hiệu địa chỉ của byte thấp là WORD1
Còn ký hiệu địa chỉ của byte cao là WORD1+1 .
Chuỗi các ký tự ( character strings)
Một mảng các mã ASCII có thể được định nghĩa bằng một chuỗi các ký tự .
Ví dụ : LETTERS DW 41h,42h,43h
tương đương với
LETTERS DW 'ABC '
Bên trong một chuỗi , ASM sẽ phân biệt chữ hoa và chữ thường . Vì vậy chuỗi 'abc' sẽ được chuyển thành 3 bytes : 61h, 62h và 63h.
Ta cũng có thể tổ hợp các ký tự và các số trong một định nghĩa.
Ví dụ : MSG DB 'HELLO', 0AH, 0DH, '$'
tương đương với
MSG DB 48H,45H,4CH,4Ch,4FH,0AH,0DH,24H
3.4 Các hằng ( constants)
Trong một chương trình các hằng có thể được đặt tên nhờ chỉ dẫn EQU (equates). Cú pháp của EQU là :
NAME EQU constant
Ví dụ : LF EQU 0AH
sau khi có khai báo trên thì LF được dùng thay cho 0Ah trong chương trình. Vì vậy ASM sẽ chuyễn các lệnh :
MOV DL,0Ah
và MOV DL,LF
thành cùng một mã máy.
Cũng có thể dùng EQU để định nghĩa một chuỗi
Ví dụ: PROMPT EQU 'TYPE YOUR NAME '
Sau khi có khai báo này, thay cho
MSG DB 'TYPE YOUR NAME '
chúng ta có thể viết
MSG DB PROMPT
3.5. Các lệnh cơ bản
CPU 8086 có rất nhiều lệnh, trong chương này, chúng ta sẽ xem xét 7 lệnh đơn giản của 8086 mà chúng thường được dùng với các thao tác di chuyển số liệu và thực hiện các phép toán số học.
Trong phần sau đây, WORD1 và WORD2 là các biến kiểu word, BYTE1 và BYTE2 là các biến kiểu byte .
3.5.1 Lệnh MOV và XCHG
Lệnh MOV dùng để chuyển số liệu giữa các thanh ghi, giữa 1 thanh ghi và 1 vị trí nhớ hoặc để di chuyển trực tiếp một số đến một thanh ghi hoặc một vị trí nhớ.
Cú pháp của lệnh MOV là :
MOV Destination , Source
Ví dụ : MOV AX,WORD1
; lấy nội dung của WORD1 đưa vào thanh ghi AX
MOV AX,BX
; AX lấy nội dung của BX, BX không thay đổi
MOV AH,'A'
; AX lấy giá trị 41h
! Chú ý lệnh MOV
Lệnh Mov không làm ảnh hưởng thanh ghi cờ hiệu
Không thể chuyển dữ liệu trực tiếp giữa 2 toán hạng mà phải dùng 1 thanh ghi trung gian
Ví dụ: Chuyển dữ liệu 16bit từ Var1 vào Var2
Mov AX,Var1
Mov Var2,AX
Không thể chuyển trực tiếp 1 hằng vào 1 thanh ghi đoạn, muốn chuyển ta phải dùng 1 thanh ghi trung gian
Ví dụ: Muốn chuyển giá trị B800h vào thanh ghi DS
Mov AX,0B800h
Mov DS,AX
Không thề chuyển trực tiếp giữa hai thanh ghi đoạn
Lệnh XCHG
Lệnh XCHG ( Exchange) dùng để trao đổi nội dung của 2 thanh ghi hoặc của một thanh ghi và một ô nhớ.
Ví dụ : XCHG AH,BL
XCHG AX,WORD1
; trao đổi nội dung của thanh ghi AX và từ nhớ WORD1.
Không thể dùng lệnh này với các thanh ghi đoạn
3.5.2 Lệnh ADD, SUB, INC, DEC
Lệnh ADD và SUB được dùng để cộng và trừ nội dung của 2 thanh ghi, của một thanh ghi và một vị trí nhớ, hoặc cộng (trừ) một số với một thanh ghi hoặc một vị trí nhớ.
Cú pháp:
ADD Destination , Source
SUB Destination , Source
Ví dụ :
ADD WORD1, AX
ADD BL , 5
SUB AX,DX ; AX=AX-DX
Không thể cộng trực tiếp thanh ghi đoạn
Lệnh ADD, SUB
Không được phép cộng hoặc trừ trực tiếp giữa 2 vị trí nhớ.
Để giải quyết vấn đề này người ta phải di chuyển byte (word) nhớ đến một thanh ghi sau đó mới cộng hoặc trừ thanh ghi này với một byte (word) nhớ khác.
Ví dụ:
MOV AL, BYTE2
ADD BYTE1, AL
Lệnh INC, DEC
Lệnh INC (incremrent) để cộng 1 vào nội dung thanh ghi hoặc một vị trí nhớ.
Lệnh DEC (decrement) để giảm bớt 1 khỏi một thanh ghi hoặc 1 vị trí nhớ.
Cú pháp:
INC Destination
DEC Destination
Ví dụ :
INC WORD1
INC AX
DEC BL
3.5.3 Lệnh NEG ( negative)
Lệnh NEG để đổi dấu của một thanh ghi hoặc một vị trí nhớ.
Cú pháp :
NEG destination
Công dụng để thực hiện phép trừ 1 hằng với 1 toán hạng (hằng không thể đứng trước). Ví dụ:
SUB 100,AL ; ASM không cho phép nên ta viết lại như sau:
SUB AL,100
NEG AL
3.6 Chuyển ngôn ngữ cấp cao
thành ngôn ngữ ASM
Giả sử A, B là 2 biến word. Anh/chị chuyển các mệnh đề sau ra ngôn ngữ ASM.
1.6.1 Mệnh đề B=A
MOV AX,A ; đưa A vào AX
MOV B,AX ; đưa AX vào B
1.6.2 Mệnh đề A=5-A
MOV AX,5 ; đưa 5 vào AX
SUB AX,A ; AX=5-A
MOV A,AX ; A=5-A
cách khác : NEG A ;A=-A
ADD A,5 ;A=5-A
1.6.3 Mệnh đề A=B-2*A
MOV AX,B ;Ax=B
SUB AX,A ;AX=B-A
SUB AX,A ;AX=B-2*A
MOV A,AX ;A=B-2*A
3.7 Cấu trúc của một
chương trình hợp ngữ
Một chương trình ngôn ngữ máy bao gồm:
mã (code)
số liệu (data)
ngăn xếp (stack).
Mỗi một phần chiếm một đoạn bộ nhớ. Mỗi một đoạn chương trình là được chuyển thành một đoạn bộ nhớ bởi ASM.
3.7.1 Các kiểu bộ nhớ
(memory models)
Độ lớn của mã và số liệu trong một chương trình được quy định bởi chỉ dẫn MODEL nhằm xác định kiểu bộ nhớ dùng với chương trình.
Cú pháp của chỉ dẫn MODEL như sau :
.MODEL memory_model
Bảng sau cho thấy các kiểu bộ nhớ :
3.7.2 Đoạn số liệu
Đoạn số liệu của chương trình chưá các khai báo biến, khai báo hằng...Để bắt đầu đoạn số liệu chúng ta dùng chỉ dẫn DATA với cú pháp như sau :
.DATA
;khai báo tên các biến , hằng và mảng
Ví dụ :
.DATA
WORD1 DW 2 ; gán 2 vào word1
WORD2 DW 5
MSG DB 'THIS IS A MESSAGE '
MASK EQU 10010010B ; hằng
3.7.3 Đoạn ngăn xếp
Mục đích của việc khai báo đoạn ngăn xếp là dành một vùng nhớ (vùng stack) để lưu trữ cho stack.
Cú pháp:
.STACK size
nếu không khai báo size thì 1KB được dành cho vùng stack.
.STACK 100h
;dành 256 bytes (=100h) cho vùng stack
3.7.4 Đoạn mã
Bắt đầu đoạn mã chưá các lệnh của chương trình:
.CODE
Bên trong đoạn mã các lệnh thường được tổ chức thành thủ tục (procedure), có cấu trúc như sau:
name PROC
;thân của thủ tục
name ENDP
Cấu trúc của 1 chương trình, phần CODE là thủ tục có tên MAIN:
.MODEL SMALL
.STACK 100h
.DATA
.CODE ; định nghĩa số liệu tại đây
MAIN PROC
;thân của thủ tục MAIN
MAIN ENDP
; các thủ tục khác nếu có
END MAIN
3.8 Các lệnh vào ra
Lệnh INT (interrupt)
Để gọi các chương trình con của BIOS và DOS có thể dùng lệnh INT với cú pháp như sau :
INT interrupt_number
ở đây interrupt_number là một số mà nó chỉ định một routine.
Ví dụ: INT 16h
; gọi routine thực hiện việc nhập số liệu từ Keyboard.
3.8.1 Lệnh INT 21h
INT 21h dùng để gọi một số lớn các các hàm (function) của DOS. Tuỳ theo giá trị mà chúng ta đặt vào thanh ghi AH, INT 21h sẽ gọi chạy một routine tương ứng .
Trong phần này ta sẽ quan tâm đến 2 hàm:
FUNCTION NUMBER ROUTINE
1 Nhập từ bàn phím một ký tự
2 Xuất ra màn hình 1 ký tự
FUCNTION 1 :
Nhập từ bàn phím một ký tự
Input : AH=1
Output: AL= mã ASCII của ký tự vừa nhập
AL=0 nếu không nhấn phím
Để gọi routine này thực hiện các lệnh sau:
MOV AH,1 ; input key function
INT 21h ; AL chứa mã ASCII của ký tự vừa nhập và hiển thị ra màn hình
FUNCTION 2 :
Xuất ra màn hình 1 ký tự
Input : AH=2
DL= mã ASCII của ký tự cần xuất
Output: AL= mã ASCII của ký tự cần xuất
Ví dụ: MOV AH,2
MOV DL,'?' ; ký tự '?'
INT 21H ; hiển thị ký tự '?'
Nếu DL chưá ký tự điều khiển thì khi gọi INT 21h, ký tự điều khiển sẽ được thực hiện. Các ký tự điều khiển thường dùng xem phụ lục A/223
ASCII code (Hex) FUNCTION
7 beep
8 b.space - nhập b.phím 1 ký tự 0 h.thị
9 tab - xuất một chuỗi ký tự ra mhình
A line feed - nhập chuỗi từ bàn phím
D carriage return (Enter)
3.9 Chương trình ví dụ: chương trình đọc một ký tự từ bàn phím và in nó trên đầu dòng mới
.MODEL SMALL
.STACK 100H
.CODE
MAIN PROC
; display dấu nhắc
MOV AH,2
MOV DL,'?'
INT 21H
; nhập 1 ký tự
MOV AH,1 ; hàm đọc ký tự
INT 21H ; ký tự được đưa vào AL
MOV BL,AL ; cất ký tự trong BL
; nhảy đến dòng mới
MOV AH,2 ; hàm xuất 1 ký tự
MOV DL,0DH ; ký tự carriage return
INT 21H ; thực hiện c.r.
MOV DL,0AH ; ký tự line feed
INT 21H ; thực hiện line feed
; xuất ký tự
MOV DL,BL ; đưa ký tự vào DL
INT 21H ; xuất ký tự
; trở về DOS
MOV AH,4CH ; hàm thoát về DOS
INT 21H ; exit to DOS
MAIN ENDP
END MAIN
3.10 Xuất một chuỗi ký tự
Trong chương trình trên đây ta đã dùng INT 21H hàm 2 và 4 để đọc và xuất một ký tự.
Hàm 9 ngắt 21H có thể dùng để xuất một chuỗi ký tự.
Input : DX=địa chỉ của chuỗi
Chuỗi kết thúc bằng dấ '$'
Ký tự $ ở cuối chuỗi sẽ không được in lên màn hình. Nếu chuỗi có chứa ký tự điều khiển thì chức năng tương ứng sẽ thực hiện.
Chương trình in lên màn hình chuỗi "HELLO!". Chuỗi HELLO được định nghĩa như sau:
MSG DB 'HELLO!$'
Lệnh LEA (Load Effective Address) nạp địa chỉ
LEA destnation , source
Ngắt 21h, hàm số 9 sẽ xuất một chuỗi ký tự ra màn hình với điều kiện địa chỉ của biến chuỗi phải ở trên DX . Có thể thực hiện điều này bởi lệnh :
LEA DX,MSG
; đưa địa chỉ offset của biến MSG vào DX
PRINT STRING PROGRAM
.MODEL SMALL
.STACK 100H
.DATA
MSG DB 'HELLO!$'
.CODE
MAIN PROC
; initialize DS
MOV AX,@DATA
MOV DS,AX
; display message
LEA DX,MSG
MOV AH,9
INT 21H
; return to DOS
MOV AH,4CH
INT 21H
MAIN ENDP
END MAIN
CASE COVERT PROGRAM
.MODEL SMALL
.STACK 100H
.DATA
CR EQU 0DH
LF EQU 0AH
MSG1 DB 'ENTER A LOWER CASE LETTER:$'
MSG2 DB 0DH,0AH,'IN UPPER CASE IT IS :'
CHAR DB ?,'$' ; định nghĩa biến CHAR có giá trị ban đầu chưa xác định
.CODE
MAIN PROC
; INITIALIZE DS
MOV AX,@DATA
MOV DS,AX
;PRINT PROMPT USER
LEA DX,MSG1 ; lấy thông điệp số 1
MOV AH,9
INT 21H ; xuất nó ra màn hình
;nhập vào một ký tự thường và đổi nó thành ký tự hoa
MOV AH,1 ; nhập vào 1 ký tự
INT 21H ; cất nó trong AL
SUB AL,20H ; đổi thành chữ hoa và cất nó trong AL
MOV CHAR, AL ; cất ký tự trong biến CHAR
; xuất ký tự trên dòng tiếp theo
LEA DX, MSG2 ; lấy thông điệp thứ 2
MOV AH,9
INT 21H ; xuất chuỗi 2, vì MSG2 0 kết thúc bởi ký tự $ nên tiếp tục xuất ký tự có trong biến CHAR
;dos exit
MOV AH,4CH
INT 21H ; dos exit
MAIN ENDP
END MAIN
Yêu cầu nhập 1 ký tự, sau đó thay đổi ký tự đã nhập thành ký tự đi liền trước theo thứ tự ASCII và in ra màn hình
3.1 Ví dụ về lệnh nhảy
Để hình dung được lệnh nhảy làm việc như thế nào chúng ta hãy viết chương trình in ra toàn bộ tập các ký tự IBM .
.MODEL SMALL
.STACK 100H
.CODE
MAIN PROC
MOV AH,2 ; hàm xuất ký tự
MOV CX,256 ; số ký tự cần xuất
MOV DL,0 ; DL giữ mã ASCII của ký tự NUL
PRINT_LOOP :
INT 21H ;display character
INC DL
DEC CX
JNZ PRINT_LOOP ;nhảy đến print_loop nếu CX# 0
;DOS EXIT
MOV AH,4CH
INT 21H
MAIN ENDP
END MAIN
Nhảy có dấu
SYMBOL DESCRITION CONDITION FOR JUMPS
JG/JNLE jump if greater than ZF=0 and SF=OF
jump if not less than or equal to
JGE/JNL jump if greater than or equal to SF=OF
jupm if not less or equal to
JL/JNGE jump if less than
jump if not greater or equal SF<>OF
JLE/JNG jump if less than or equal ZF=1 or SF<>OF jump if not greater
Lệnh CMP ( Compare)
Các lệnh nhảy thường lấy kết qủa của lệnh Compare như là điều kiện. Cú pháp của lệnh CMP là :
CMP destination, source
Lệnh này so sánh toán hạng nguồn và toán hạng đích bằng cách tính hiệu Dest - Src. Kết qủa không được cất giữ . Lệnh CMP giống như lệnh SUB, chỉ khác: lệnh CMP toán hạng đích không thay đổi.
Hiển thị câu hỏi "Is if after 12 noon (Y/N)? và chờ ta bấm một phím,
nếu ta bấm y hay Y thông báo "Good afternoon!",
còn bấm n hay N (hay bất kỳ phím) thông báo "Good morning!"
.MODEL SMALL
.STACK 100h
.DATA
TimePrompt DB 'Is it after 12 noon (Y/N)?$
GoodMorningMsg LABEL BYTE
DB 0Dh,0Ah, 'Good morning!',0Dh,0Ah,'$'
GoodAfternoonMsg LABEL BYTE
DB 0Dh,0Ah, 'Good afternoon!',0Dh,0Ah,'$'
.CODE
Begin:
MOV AX, @DATA
MOV DS, AX
LEA DX,TimePrompt
MOV AH, 09h
INT 21h
MOV AH, 01h ;Nhập một ký tự chứa vào AL
INT 21h
CMP AL, 'y' ; Ký tự đã gõ là y?
JZ IsAfternoon ; Đúng, nhảy tới IsAfternoon
CMP AL, 'Y' ;Ký tự gõ là Y?
JNZ IsMorning ;Không nhảy tới IsMorning
IsAfternoon:
;Không gõ y hay Y
LEA DX,GoodAfternoonMsg
JMP DisplayGreeting
IsMorning:
LEA DX,GoodMorningMsg
DisplayGreenting:
MOV AH, 09h
INT 21h
MOV AX,4C00h
INT 21h
END Begin
Diễn dịch lệnh nhảy có điều kiện
Ví dụ trên đây về lệnh CMP cho phép lệnh nhảy sau nó chuyển điều khiển đến nhãn BELOW các lệnh
CMP AX,BX
JG BELOW
có nghĩa là nếu AX>BX thì nhảy đến nhãn BELOW
Mặc dù lệnh CMP được thiết kế cho các lệnh nhảy. Nhưng lệnh nhảy có thể đứng trước 1 lệnh khác, chẳng hạn :
DEC AX
JL THERE
có nghĩa là nếu AX trong diễn dịch có dấu < 0 thì điều khiển được chuyển cho THERE
3.11 Lệnh JMP
Lệnh JMP ( jump) là lệnh nhảy không điều kiện.
Cú pháp:
JMP destination
Trong đó destination là một nhãn ở trong cùng 1 đọan với lệnh JMP.
Lệnh JMP dùng để khắc phục hạn chế của các lệnh nhảy có điều kiện (không quá 126 bytes kể từ vị trí của lệnh nhảy có điều kiện)
Ví dụ chúng ta có đoạn chương trình sau :
TOP:
; thân vòng lặp
DEC CX
JNZ TOP ; nếu CX>0 tiếp tục lặp
MOV AX,BX
giả sử thân vòng lặp chứa nhiều lệnh mà nó vượt khỏi 126 bytes trước lệnh JNZ TOP . Có thể giải quyết tình trạng này bằng các lệnh sau :
TOP:
; thân vòng lặp
DEC CX
JNZ BOTTOM ; nếu CX>0 tiếp tục lặp
JMP EXIT
BOTTOM:
JMP TOP
EXIT:
MOV AX,BX
3.12 Cấu trúc rẽ nhánh
Trong ngôn ngữ cấp cao cấu trúc rẽ nhánh cho phép một chương trình rẽ nhánh đến những đoạn khác nhau tuỳ thuộc vào các điều kiện . Trong phần này chúng ta sẽ xem xét 3 cấu trúc
a. IF-THEN
b. IF_THEN_ELSE
c. CASE
a. IF-THEN
Cấu trúc IF-THEN có thể diễn đạt như sau :
IF condition is true
THEN
execute true branch statements
END IF
Ví dụ : Thay thế giá trị trên AX bằng giá trị tuyết đối của nó
IF AX<0
THEN
replace AX by -AX
END-IF
Có thể mã hoá như sau :
; if AX<0
CMP AX,0
JNL END_IF ; no, exit
;then
NEG AX ; yes, đổi dấu
END_IF :
b. IF_THEN_ELSE
IF condition is true
THEN
execute true branch statements
ELSE
execute false branch statements
END_IF
Ví dụ : giả sử AL và BL chứa ASCII code của 1 ký tự. Hãy xuất ra màn hình ký tự trước (theo thứ tự ký tự ). Thuật toán
IF AL<= BL
THEN
display AL
ELSE
display character in BL
END_IF
Có thể mã hoá như sau :
MOV AH,2
; chuẩn bị xuất ký tự
;if AL<=BL
CMP AL,BL ;AL<=BL?
JNBE ELSE_
; nếu không nhỏ hơn hay = BL
;then
MOV DL,AL
JMP DISPLAY
ELSE_:
MOV DL,BL
DISPLAY:
INT 21H
END_IF :
c. CASE
Case là một cấu trúc rẽ nhánh nhiều hướng. Cấu trúc của CASE như sau :
CASE expression
value_1 : Statements_1
value_2 : Statements_2
...
value_n : Statements_n
Ví dụ : Nếu AX < 0 thì đặt -1 vào BX
Nếu AX = 0 thì đặt 0 vào BX
Nếu AX > 0 thì đặt 1 vào BX
Thuật toán :
CASE AX
< 0 put -1 in BX
= 0 put 0 in BX
> 0 put 1 in BX
Có thể mã hoá như sau :
; case AX
CMP AX,0 ;test AX
JL NEGATIVE ;AX<0
JE ZERO ;AX=0
JG positive ;AX>0
NEGATIVE:
MOV BX,-1
JMP END_CASE
ZERO:
MOV BX,0
JMP END_CASE
POSITIVE:
MOV BX,1
JMP END_CASE
END_CASE :
Rẽ nhánh với tổ hợp các điều kiện
Đôi khi tình trạng rẽ nhánh trong các lệnh IF ,CASE cần một tổ hợp các điều kiện dưới dạng :
Condition_1 AND Condition_2
Condition_1 OR Condition_2
Ví dụ về điều kiện AND :
Đọc 1 ký tự và nếu là ký tự hoa thì in ra MH.
Thuật toán :
Read a character ( into AL)
IF ( 'A'<= character ) AND ( charater <= 'Z')
THEN
display character
END_IF
Ví dụ về điều kiện AND :
Đọc 1 ký tự và nếu là ký tự hoa thì in ra MH.
; read a character
MOV AH,2
INT 21H ; character in AL
; IF ( 'A'<= character ) AND ( charater <= 'Z')
CMP AL,'A' ; char >='A'?
JNGE END_IF ;no, exit
CMP AL,'Z' ; char <='Z'?
JNLE END_IF ; no exit
; then display it
MOV DL,AL
MOV AH,2
INT 21H
END_IF :
Ví dụ về điều kiện OR : Đọc một ký tự , nếu ký tự đó là 'Y' hoặc 'y' thì in nó lên màn hình , ngược lại thì kết thúc chương trình .
Thuật toán
Read a charcter ( into AL)
IF ( chr ='Y') OR ( chr='y')
THEN
dispplay it
ELSE
terminate the program
END_IF
Code
;read a character
MOV AH,2
INT 21H
; character in AL
;IF ( chr ='y' ) OR ( chr = 'Y')
CMP AL,'y' ; chr ='y'?
JE THEN ;=, in Chr
CMP AL,'Y' ; chr ='Y'?
JE THEN ;=, in Chr
JMP ELSE_ ;<> , terminate
THEN :
MOV DL,AL
MOV AH,2
INT 21H
JMP END_IF
ELSE_:
MOV AH,4CH
INT 21h
END_IF :
3.13 Cấu trúc lặp
Một vòng lặp gồm nhiều lệnh được lặp lại, số lần lặp phụ thuộc điều kiện.
a. Vòng FOR
b. Vòng WHILE
c. Vòng REPEAT
a. Vòng FOR
Lệnh LOOP có thể dùng để thực hiện vòng FOR.
Cú pháp :
LOOP destination_label
Số đếm cho vòng lặp là thanh ghi CX mà ban đầu nó được gán 1 giá trị nào đó . Khi lệnh LOOP được thực hiện CX sẽ tự động giảm đi 1. Nếu CX chưa bằng 0 thì vòng lặp được thực hiện tiếp tục. Nếu CX=0 lệnh sau lệnh LOOP được thực hiện
Dùng lệnh LOOP , vòng FOR có thể thực hiện như sau :
; gán cho cho CX số lần lặp
TOP:
; thân của vòng lặp
LOOP TOP
Ví dụ : Dùng vòng lặp in ra 1 hàng 80 dấu '*'
MOV CX,80 ; CX chưá số lần lặp
MOV AH,2 ; hàm xuất ký tự
MOV DL,'*' ;DL chưá ký tự '*'
TOP:
INT 21h ; in dấu '*'
LOOP TOP ; lặp 80 lần
Lưu ý rằng vòng FOR cũng như lệnh LOOP thực hiện ít nhất là 1 lần. Do đo nếu ban đầu CX=0 thì vòng lặp sẽ thực hiện lặp đến 65535 lần. Để tránh tình trạng này, lệnh JCXZ (Jump if CX is zero) phải được dùng trước vòng lặp.
Lệnh JXCZ có cú pháp như sau :
JCXZ destination_label
Nếu CX=0 điều khiển được chuyển cho destination_label. Các lệnh sau đây sẽ đảm bảo vòng lặp không thực hiện nếu CX=0
JCXZ SKIP
TOP :
; thân vòng lặp
LOOP TOP
SKIP :
b. Vòng WHILE
Vòng WHILE phụ thuộc vào 1 điều kiện. Nếu điều kiện đúng thì thực hiện vòng WHILE. Vì vậy nếu điều kiện sai thì vòng WHILE không thực hiện gì cả .
Ví dụ : Viết đoạn mã đếm số ký tự được nhập vào trên cùng một hàng .
MOV DX,0 ; DX để đếm số ký tự
MOV AH,1 ;hàm đọc 1 ký tự
INT 21h ; đọc ký tự vào AL
WHILE_:
CMP AL,0DH ; có phải là ký tự CR?
JE END_WHILE ; đúng , thoát
INC DX ;tăng DX lên 1
INT 21h ; đọc ký tự
JMP WHILE_ ; lặp
END_WHILE :
c. Vòng REPEAT
Cấu trúc REPEAT: repeat statements
until condition
Trong cấu trúc repeat mệnh đề được thi hành đồng thời điều kiện được kiểm tra. Nếu điều kiện đúng thì vòng lặp kết thúc.
Ví dụ : Viết đoạn mã đọc vào các ký tự cho đến khi gặp ký tự trống .
MOV AH,1 ; đọc ký tự
REPEAT:
INT 21h ; ký tự trên AL
;until
CMP AL,' ' ; AL=' '?
JNE REPEAT
Lưu ý: REPEAT tiến hành ít nhất 1 lần, trong khi đó WHILE có thể không tiến hành lần nào nếu từ đầu điều kiện bị sai.
3.14 Lập trình với cấu trúc cấp cao
Bài toán : Viết chương trình nhắc người dùng gõ vào một dòng văn bản. Trên 2 dòng tiếp theo in ra ký tự viết hoa đầu tiên và ký tự viết hoa cuối cùng theo thứ tự alphabetical. Nếu người dùng gõ vào một ký tự thường, máy sẽ thông báo 'No capitals'
Kết qủa chạy chương trình sẽ như sau :
Type a line of text :
TRUONG DAI HOC SU PHAM
First capital = A
Last capital = U
Để giải bài toán này ta dùng kỹ thuật lập trình TOP-DOWN, nghĩa là chia nhỏ bài toán thành nhiều bài toán con. Có thể chia bài toán thành 3 bài toán con như sau :
1. Xuất 1 chuỗi ký tự ( lời nhắc)
2. Đọc và xử lý 1 dòng văn bản
3. In kết qủa
Bước 1: Hiện dấu nhắc .
Bước này có thể mã hoá như sau :
MOV AH,9 ; hàm xuất chuỗi
LEA DX,PRMOPT ;lấy địa chỉ chuỗi vào DX
INT 21H ; xuất chuỗi
Dấu nhắc có thể mã hoá như sau trong đoạn số liệu .
PROMPT DB 'Type a line of text:',0DH,0AH,'$'
Bước 2 : Đọc và xử lý một dòng văn bản
Đọc các ký tự từ bàn phím , tìm ra ký tự đầu và ký tự cuối , nhắc nhở người dùng nếu ký tự gõ vào không phải là ký tự hoa.
Có thể biễu diễn bước này bởi thuật toán sau :
Read a character
WHILE character is not a carrige return DO
IF character is a capital (*)
THEN
IF character precedes first capital
Then
first capital= character
End_if
IF character follows last character
Then
last character = character
End_if
END_IF
Read a character
END_WHILE
Trong đó dòng (*) có nghĩa là điều kiện để ký tự là hoa là điều kiện AND
IF ( 'A'<= character ) AND ( character <= 'Z')
MOV AH,1 ; đọc ký tự
INT 21H ; ký tự trên AL
WHILE :
;trong khi ký tự gõ vào không phải là CR thì thực hiện
CMP AL,0DH ; CR?
JE END_WHILE ;yes, thoát
; nếu ký tự là hoa
CMP AL,'A' ; char >='A'?
JNGE END_IF
;không phải ký tự hoa thì nhảy đến END_IF
CMP AL,'Z' ; char <= 'Z'?
JNLE END_IF
; không phải ký tự hoa thì nhảy đến END_IF
; thì
; nếu ký tự nằm trước biến FIRST (giá trị ban đầu là'[' : ký tự sau Z )
CMP AL,FISRT ; char
JNL CHECK_LAST; >=
; thì ký tự viết hoa đầu tiên = ký tự
MOV FIRST,AL ;FIRST=chr.
;end_if
CHECK_LAST:
; nếu ký tự là sau biến LAST ( giá trị ban đầu là '@': ký tự trước A)
CMP AL,LAST ; char > LAST ?
JNG END_IF ; <=
;thì ký tự cuối cùng = ký tự
MOV LAST, AL ;LAST = character
;end_if
END_IF :
; đọc một ký tự
INT 21H ; ký tự trên AL
JMP WHILE_ ; lặp
END_WHILE:
Các biến FIRST và LAST định nghĩa như sau:
FIRST DB '[ $' ;'[' là ký tự sau Z
LAST DB '@ $ ' ;'@' là ký tự trước A
Bước 3 : In kết qủa
Thuật toán
IF no capital were typed
THEN
display 'No capital'
ELSE
display first capital and last capital
END_IF
Bước 3 sẽ phải in ra các thông báo :
NOCAP_MSG nếu không phải chữ in
CAP1_MSG chữ in đầu tiên
CAP2_MSG chữ in cuối cùng
Chúng được định nghĩa như sau trong đoạn số liệu .
NOCAP_MSG DB 0DH,0AH,'No capitals $'
CAP1_MSG DB 0DH,0AH, 'First capital= '
FIRST DB '[ $ '
CAP2_MSG DB 0DH,0AH,'Last capital='
LAST DB '@ $'
Bước 3 có thể mã hoá như sau :
;in kết quả
MOV AH,9 ; hàm xuất ký tự
; IF không có chữ hoa nào được nhập thì FIRST ='['
CMP FIRST,'[' ; FIRST='[' ?
JNE CAPS ; không , in kết qủa
;THEN
LEA DX,NOCAP_MSG
INT 21H
CAPS:
LEA DX,CAP1_MSG
INT 21H
LEA DX,CAP2_MSG
INT 21H
; end_if
Chương trình có thể viết như sau:
.MODEL SMALL
.STACK 100h
.DATA
PROMPT DB 'Type a line of text', 0DH, AH, '$'
NOCAP_MSG DB 0DH,0AH, 'No capitals $'
CAP1_MSG DB 0DH,0AH, 'First capital='
FIRST DB '[ $'
CAP2_MSG DB 'Last capital = '
LAST DB '@ $'
.CODE
MAIN PROC
; khởi tạo DS
MOV AX,@DATA
MOV DS,AX
; in dấu nhắc
MOV AH,9 ; hàm xuất chuỗi
LEA DX,PROMPT ;lấy địa chỉ chuỗi vào DX
INT 21H ; xuất chuỗi
;đọc và xử lý 1 dòng văn bản
MOV AH,1 ; đọc ký tự
INT 21H ; ký tự trên AL
WHILE :
;trong khi ký tự gõ vào không phải là CR thì thực hiện
CMP AL,0DH ; CR?
JE END_WHILE ;yes, thoát
; nếu ký tự là hoa
CMP AL,'A' ; char >='A'?
JNGE END_IF ;0 phải K.tự hoa, nhảy đến END_IF
CMP AL,'Z' ; char <= 'Z'?
JNLE END_IF ; 0 phải k.tự hoa, nhảy đến END_IF
; thì
; nếu ký tự nằm trước biến FIRST
CMP AL,FISRT ; char < FIRST ?
JNL CHECK_LAST ; >=
; thì ký tự viết hoa đầu tiên = ký tự
MOV FIRST,AL ; FIRST=character
;end_if
CHECK_LAST:
; nếu ký tự là sau biến LAST
CMP AL,LAST ; char > LAST ?
JNG END_IF ; <=
;thì ký tự cuối cùng = ký tự
MOV LAST, AL ;LAST = character
;end_if
END_IF :
; đọc một ký tự
INT 21H ; ký tự trên AL
JMP WHILE_ ; lặp
END_WHILE:
;in kết quả
MOV AH,9 ; hàm xuất ký tự
; IF không có chữ hoa nào được nhập thì FIRST ='['
CMP FIRST,'[' ; FIRST='[' ?
JNE CAPS ; không , in kết qủa
;Then
LEA DX,NOCAP_MSG
INT 21H
CAPS:
LEA DX,CAP1_MSG
INT 21H
LEA DX,CAP2_MSG
INT 21H
; end_if
; dos exit
MOV AH,4CH
INT 21h
MAIN ENDP
END MAIN
Bài tập1:
Viết chương trình nhập vào 1 số (2-8)
In ra màn hình ngày trong tuần tương ứng
Bài tập2:
Nhập vào 1 số nguyên n, không dấu (1-10)
Tính tổng=1+2+..+N
Mul desc
Desc là thanh ghi bất kỳ, nhân với giá trị trong thanh ghi AL
Kết quả lưu vào AX
Bạn đang đọc truyện trên: AzTruyen.Top