「みんな教えて! 選手権!!」開催のお知らせ

下記の2個のファイルを持つプログラムでインラインアセンブラのcall命令で配列変数のポインター(アドレス)が勝手に変わる現象をおしえてください。
但し、配列の最初のポインターのみが変わる。
開発環境はWin7(64bit)、VC++2010無償版です。

main.cpp

int *disp;
int data[3];

WinMain()
{
_asm mov disp,offset disp_top



メッセージループへ

_asm{
disp_top://下記move()からcallされた時
lea eax,data//下記のmove()関数のeaxの値より16番地少ない
lea ebx,data+4//下記のmove()関数のebxの値と同じ
lea ecx,data+8//下記のmove()関数のecxの値と同じ



ret
  }
}



move.cpp

extern int *disp;
extern int data[3];

void move()
{
_asm{
lea eax,data
lea ebx,data+4
lea ecx,data+8
call disp
  }
}

A 回答 (3件)

Win32 プロジェクトでやってみた.


wmem.cpp
// wmem.cpp : アプリケーションのエントリ ポイントを定義します。
//

#include "stdafx.h"
#include "wmem.h"

int *disp;
int data[3];
void foo();

int APIENTRY _tWinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPTSTR lpCmdLine,
int nCmdShow)
{
_asm mov disp, offset disp_top

foo();
goto fin;

_asm {
disp_top:
lea ecx, data
lea edx, data+4
ret
}

fin:
return 0;
}

wmemsub.cpp
#include "stdafx.h"

extern int *disp;
extern int data[3];

void foo()
{
_asm {
lea eax, data
lea ebx, data+4
call disp
}
}

works as expected.
    • good
    • 0
この回答へのお礼

再度ご提示頂きありがとうございます。
オプションでコンパイル済みヘッダーstdafx.hを使う設定にしたのですが、ヘッダーファイルが開けないようです。
wmem.hについてもwindows(x86)やProgram Filesフォルダーを検索したのですが見当たらず、コンパイラーでこの2つのヘッダーが開かないエラーとなりました。
せっかくコードを示して頂いたのですが残念です。

そこで私の開発環境に同じプログラムを移植してeax,ebx,ecx,edxの内容を確認しましたら、思った通りeax=ecx,ebx=edxとなりました。
変数の数が少ないとOKのようです。

ただ、気がかりなのは6から2010に移植した時、ソーステキスト表示のタブの所に空白が表示されず、くっ付いた表示となりました。
例えばcall dispがcalldispとなりました。
ところがlとdの間のタブコードは存在するため、このままでもコンパイルエラーは発生しませんでした。表示上の問題だと考えますが、これも不可解な現象です。プログラムが大き過ぎたのでしょうか?この不具合を検証するため、プログラムをスリムにして行く途中でタブの所に空白が表示されるようになりました。

やはり大きなプログラム(行数)で質問のような問題が発生するようです。
シンドイですが、プログラムを細分化するか、自力でバグを見つける以外に方法はないようです。

本当に色々とお世話になり感謝しています。ありがとうございました。

お礼日時:2011/05/19 17:53

ソースを見て「main があって WinMain がない」なら「それはコンソールアプリケーションである」とくらい判断できない?

この回答への補足

コンソールで立ち上げると、ビルドできました。
しょせん私のスキルはこの程度ですので、そこのところ、よろしくお願いします。
只今、小さなプログラムで確認するべく問題のソースを出来る限り短縮しております。
その内にまた、よろしくお願いします。

出来ましたら、複雑になりますがWinMainでの検証をして頂きますと、ありがたいのですが・・・

補足日時:2011/05/18 18:34
    • good
    • 0

とりあえず手元で


memmain.cpp
#include <cstdio>

int *disp;
int data[3];

void foo();

int main()
{
_asm mov disp, offset disp_top

foo();
goto fin;

_asm {
disp_top:
lea ecx, data
lea edx, data+4
ret
}

fin:
return 0;
}

memsub.cpp
extern int *disp;
extern int data[3];

void foo()
{
_asm {
lea eax, data
lea ebx, data+4
call disp
}
}

で実験した限りでは全く問題なし (lea edx, data+4 のあとで eax == ecx, ebx == edx).
ちなみに Windows7 Ultimate (64bit), VisualStudio 2010 Express.

この回答への補足

ご提示頂いたソースをビルドしたら、次のようなエラーがでました。
手直しするところがあればお願いします。

1> コードを生成中...
1>LIBCMT.lib(wincrt0.obj) : error LNK2001: 外部シンボル "_WinMain@16" は未解決です。

補足日時:2011/05/18 16:40
    • good
    • 0

お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!


おすすめ情報