Шаг 90.
Вызовы API и ресурсы в ассемблерных модулях

    На этом шаге мы рассмотрим использование вызовов API и ресурсов.

    На данном шаге мы покажем, что вызываемая ассемблерная процедура может содержать не только какие-то вспомогательные процедуры, но и работу с функциями API и ресурсами. Причем ресурсы, разумеется, являются для всех модулей проекта общими. Можно иметь несколько файлов, задающих ресурсы, но главное, чтобы не совпадали имена и идентификаторы.


    Консольная программа на C++ вызывает процедуру, определенную в ассемблерном модуле, которая, в свою очередь, работает в GUI-режиме.
#include <windows.h>
#include <stdio.h>
//Объявляется внешняя функция.
extern  "C" __stdcall DIAL1();
void main()
{
   DIAL1();
   ExitProcess(0);
}

    Файл pr90_1.rc:

//Определение  констант.
#define WS_SYSMENU        0x00080000L
//Элементы на окне должны быть изначально видимы.
#define WS_VISIBLE        0x10000000L
//Бордюр вокруг элемента.
#define WS_BORDER         0x00800000L
//При помощи TAB можно по очереди активизировать элементы.
#define WS_TABSTOP        0x00010000L
//Текст в окне редактирования прижат к левому краю.
#define ES_LEFT           0x0000L
//Стиль всех элементов на окне.
#define WS_CHILD          0x40000000L
//Запрещается ввод с клавиатуры.
#define ES_READONLY       0x0800L
#define DS_3DLOOK         0x0004L
//Определение диалогового окна.
DIAL1 DIALOG 0, 0, 240, 100
STYLE WS_SYSMENU | DS_3DLOOK
CAPTION "Диалоговое окно с часами и датой"
FONT 8, "Arial"
{
  CONTROL "", 1, "edit", ES_LEFT | WS_CHILD | WS_VISIBLE | 
    WS_BORDER | WS_TABSTOP | ES_READONLY, 100, 5, 130, 12
}

    Файл pr90_1.inc:

;Константы.
;Сообщение приходит при закрытии окна.
WM_CLOSE      equ 10h
;Сообщение приходит при создании окна.
WM_INITDIALOG equ 110h
;Сообщение приходит при событии с элементом на окне.
WM_COMMAND    equ 111h
;Сообщение от таймера.
WM_TIMER      equ 113h
;Сообщение посылки текста элементу.
WM_SETTEXT    equ 0Ch
;Прототипы внешних процедур.
IFDEF MASM
    EXTERN     SendDlgItemMessageA@20:NEAR 
    EXTERN     wsprintfA:NEAR 
    EXTERN     GetLocalTime@4:NEAR 
    EXTERN     ExitProcess@4:NEAR 
    EXTERN     GetModuleHandleA@4:NEAR 
    EXTERN     DialogBoxParamA@20:NEAR
    EXTERN     EndDialog@8:NEAR 
    EXTERN     SetTimer@16:NEAR 
    EXTERN     KillTimer@8:NEAR
ELSE
    EXTERN     SendDlgItemMessageA:NEAR 
    EXTERN     _wsprintfA:NEAR 
    EXTERN     GetLocalTime:NEAR 
    EXTERN     ExitProcess:NEAR 
    EXTERN     GetModuleHandleA:NEAR 
    EXTERN     DialogBoxParamA:NEAR
    EXTERN     EndDialog:NEAR
    EXTERN     SetTimer:NEAR
    EXTERN     KillTimer:NEAR

    SendDlgItemMessageA@20 = SendDlgItemMessageA 
    wsprintfA              = _wsprintfA 
    GetLocalTime@4         = GetLocalTime 
    ExitProcess@4          = ExitProcess 
    GetModuleHandleA@4     = GetModuleHandleA 
    DialogBoxParamA@20     = DialogBoxParamA 
    EndDialog@8            = EndDialog 
    SetTimer@16            = SetTimer 
    KillTimer@8            = KillTimer 
ENDIF
;Структуры.
;Структура сообщения.
MSGSTRUCT  STRUC	
           MSHWND     DD ? ;Идентификатор окна, получающего сообщение.
           MSMESSAGE  DD ? ;Идентификатор сообщения.
           MSWPARAM   DD ? ;Доп. информация о сообщении.
           MSLPARAM   DD ? ;Доп. информация о сообщении.
           MSTIME     DD ? ;Время посылки сообщения.
           MSPT       DD ? ;Положение курсора во время посылки сообщения.
MSGSTRUCT ENDS	
;Cтруктура данных дата-время.
DAT        STRUC		
           year    DW   ?	
           month   DW   ?	
           dayweek DW   ?	
           day     DW   ?	
           hour    DW   ?	
           min     DW   ?	
           sec     DW   ?	
           msec    DW   ?	
DAT ENDS

    Файл pr90_1.asm:

.386P
;Плоская модель.
.MODEL FLAT, STDCALL
include pr90_1.inc
PUBLIC DIAL1
;------------------------------------------------
;Сегмент данных. 
_DATA SEGMENT DWORD PUBLIC USE32 'DATA'
     MSG        MSGSTRUCT  <?>
     HINST      DD 0 ;Дескриптор приложения.
     PA         DB 'DIAL1',0
     TIM        DB "Дата  %u/%u/%u Время  %u:%u:%u",0
     STRCOPY    DB  50   DUP(?) 
     DATA       DAT  <0>
_DATA ENDS 
;Сегмент кода.
_TEXT SEGMENT DWORD PUBLIC USE32 'CODE'
DIAL1  PROC
     PUSH  EBP
     MOV   EBP,ESP
;Получить дескриптор приложения.
     PUSH  0
     CALL  GetModuleHandleA@4
     MOV   [HINST], EAX
;-------------------------------------
;Создать диалоговое окно.
     PUSH  0
     PUSH  OFFSET WNDPROC
     PUSH  0
     PUSH  OFFSET PA
     PUSH  [HINST] 
     CALL  DialogBoxParamA@20 
     CMP   EAX,-1 
     POP   EBP
     RET
DIAL1  ENDP 
;-------------------------------------
;-------------------------------------
;Процедура окна.
;Расположение параметров в стеке: 
;[ЕВР+014Н] LPARAM 
;[ЕВР+10Н] WAPARAM 
;[ЕВР+0СН] MES 
;[ЕВР+8] HWND 
WNDPROC PROC
        PUSH EBP
        MOV  EBP, ESP
        PUSH EBX
        PUSH ESI
        PUSH EDI
;-------------------------------------
        CMP  DWORD PTR [EBP+0CH],WM_CLOSE
        JNE  L1
;Здесь реакция на закрытие окна. 
;Удалить таймер 1.
        PUSH 1   ;Идентификатор таймера. 
        PUSH DWORD  PTR  [EBP+08H] 
        CALL KillTimer@8 
;Удалить таймер 2.
        PUSH 2   ;Идентификатор таймера. 
        PUSH DWORD  PTR  [EBP+08H] 
        CALL KillTimer@8 
;Закрыть диалог.
        PUSH 0
        PUSH DWORD PTR [EBP+08H]
        CALL EndDialog@8
        JMP  FINISH 
L1:
        CMP  DWORD PTR [EBP+0CH],WM_INITDIALOG
        JNE  L2
;Здесь начальная инициализация. 
;Установить таймер 1.
        PUSH 0      ;Параметр = NULL 
        PUSH 1000   ;Интервал  1 с 
        PUSH 1      ;Идентификатор таймера 
        PUSH DWORD  PTR  [EBP+08H] 
        CALL SetTimer@16 
;Установить таймер 2.
        PUSH OFFSET TIMPROC  ;Параметр = NULL 
        PUSH 500             ;Интервал 0.5 с 
        PUSH 2               ;Идентификатор таймера 
        PUSH DWORD  PTR  [EBP+08H] 
        CALL SetTimer@16 
        JMP  FINISH 
L2:
        CMP  DWORD PTR [EBP+0CH],WM_TIMER 
        JNE  FINISH 
;Отправить строку в окно. 
        PUSH OFFSET  STRCOPY 
        PUSH 0
        PUSH WM_SETTEXT
        PUSH 1  ;Идентификатор элемента. 
        PUSH DWORD PTR [EBP+08H] 
        CALL SendDlgItemMessageA@20 
FINISH:
        POP  EDI
        POP  ESI
        POP  EBX
        POP  EBP
        MOV  EAX,0
        RET  16
WNDPROC ENDP
;-------------------------------------
;Процедура таймера. 
;Расположение параметров в стеке.
;[ЕВР+014Н]   LPARAM - промежуток запуска Windows.
;[ЕВР+10Н]    WAPARAM - идентификатор таймера.
;[EBP+0CH]    WM_TIMER
;[ЕВР+8]      HWND
TIMPROC PROC 
        PUSH EBP 
        MOV  EBP,ESP
;Получить локальное время. 
        PUSH OFFSET DATA 
        CALL GetLocalTime@4
;Получить  строку для вывода даты и времени. 
        MOVZX EAX, DATA.sec 
        PUSH EAX
        MOVZX EAX, DATA.min 
        PUSH EAX
        MOVZX EAX,DATA.hour 
        PUSH EAX 
        MOVZX EAX,DATA.year
        PUSH EAX
        MOVZX EAX,DATA.month
        PUSH EAX
        MOVZX EAX,DATA.day
        PUSH EAX
        PUSH OFFSET TIM
        PUSH OFFSET STRCOPY
        CALL  wsprintfA 
;Восстановить стек.
        ADD  ESP,32
        POP  EBP
        RET  16
TIMPROC ENDP 
_TEXT   ENDS	
        END 
Тексты этих файлов можно взять здесь.

    Трансляция:

    tasm32  /ml pr90_1.asm
    brcc32  pr90_1.rc

    Далее включаем в проект файлы pr90_1.obj и pr90_1.res.

    На следующем шаге мы приведем пример использования ассемблера и C.




Предыдущий шаг Содержание Следующий шаг