Microsoft MVP성태의 닷넷 이야기
mips 어셈블리 연산 [링크 복사], [링크+제목 복사]
조회: 6752
글쓴 사람
ocm
홈페이지
첨부 파일

안녕하세요 정성태 선생님, 저는 컴퓨터를 독학하는 학생입니다.
비록 c#관련 질문은 아니지만 질문드려도 괜찮을까요?

컴파일러가 프로시저를 번역하는 데 인수 레지스터가 4개, 결과값 레지스터가 2개만으로 부족한 경우 레지스터 스필링이 필요해서 스택에 레지스터들을 저장하는데요. 여기서 레지스터를 sw를 통해 레지스터에서 스택에 저장한 다음에 산술 연산이 가능한가요? 제가 알기론 mips연산은 레지스터에서만 가능한데 스택에 값을 저장하고 산술 연산이 가능한 이유를 모르겠습니다. 무슨 원리인지 가르쳐주신다면 정말 감사드립니다.








[최초 등록일: ]
[최종 수정일: 6/14/2021]


비밀번호

댓글 작성자
 



2021-06-15 04시36분
제가 MIPS 어셈블리는 잘 모릅니다. 단지 일반적인 어셈블리를 알고 있는 지식에 기준해 설명을 드리겠습니다.

우선 "레지스터를 sw를 통해 레지스터에서 스택에 저장한 다음에 산술 연산이 가능"하냐고 물은 것은 아마도 스택에 저장한 다음 필요할 때 다시 레지스터에 로드해 연산을 계속하는 식으로 하게 될 것입니다.

말씀하신 것처럼 레지스터가 총 6개(인수 4개 $a0, $a1, $a2, $a3, 결과 2개 $f0, $f1)만을 가진 MIPS CPU를 가정하는 경우, 만약 5개의 인자를 넘겨야한다면 다음과 같은 식으로 프로시저를 호출하게 될 것입니다.

$a0 = 1
$a1 = 2
$a2 = 3
$a3 = 4
push 5
call calc;

그러면 calc 안에서는 스택 공간을 하나 더 확보하는 식으로 진행하면 됩니다. 예를 들어, 레지스터가 4바이트라고 가정하면 calc 프로시저에서는 시작 단계에서 4바이트의 스택을 확보하면 됩니다.

push bp
mov bp, sp
sub sp, 4 (스택이 아래로 자란다고 가정)

이후, calc 프로시저의 5번째 인자로 전달한 값을 연산해야 하면, 그것을 로드할 레지스터를 결정해야 할 것입니다. 만약 그 레지스터를 $a3이라고 가정하면, $a3 값을 스택에 보관해 둔 후,

mov [bp - 4], $a3 // [bp - 4]는 calc가 도입 시 확보한 영역이라고 가정.

5번째로 전달한 인자를 $a3 레지스터에 로드한 후 연산을 진행하면 될 것입니다.

mov $a3, [bp + 8] // [bp + 8]에 calc로 전달한 5번째 인자가 저장되어 있다고 가정
add $a1, $a1, $a3 // $a1 = $a1 + $a3

혹시 원하는 답변이 아니면, 다시 질문해 주세요. ^^
정성태
2021-06-15 11시19분
[ocm] 답변 감사드립니다. 그런데 스택에 저장한 다음 필요할 때 레지스터에 로드해 연산을 하려면 lw를 해야 스택에서 레지스터로 로드되서 연산을 계속할 수 있는 것 아닌지가 헷갈립니다.
[guest]
2021-06-15 11시33분
당연히 lw를 해야 하지 않을까요? ^^ 위에서 답변한 내용으로 보면 "mov $a3, [bp - 8]"의 코드가 스택의 값을 $a3 레지스터로 옮기는 것입니다.
정성태
2021-06-15 12시10분
[ocm] 저도 이것이 맞다고 생각하는데 컴퓨터구조및설계(저자 david a patterson)라는 책의 예제에는 sw후 바로 연산을 하는걸로 나와있더라고요. "https://youtu.be/wUcBordHEDo"(최린 교수 컴퓨터구조 강의)에서도 29분부터 나오는 예제에서도 sw후에 바로 연산을 하시더라고요. 내용 정리해서 질문게시판에 질문 다시 올려보았습니다.
[guest]
2021-06-15 01시34분
말씀해 주신 강의를 봤는데요. 그 예제에서 sw 후에 바로 연산을 하는 것은 sw로 스택에 저장한 메모리를 대상으로 연산하는 것이 아닙니다.

sub $sp, $sp, 8
sw $t1, 4($sp)
sw $t0, 0($sp)

위의 명령어는 procedure를 위해 (4바이트 공간의) 변수 2개를 마련한 다음 현재 $t1과 $t0 레지스터에 있는 값을 스택에 보관하는 명령어입니다. 왜냐하면, $t1과 $t0 레지스터를 그 이후의 명령어에서 사용해야 하기 때문에 기존 값을 백업해 두는 것입니다.

그다음의 명령을 보면,

add $t0, $a0, $a1
add $t1, $a2, $a3

가 나오는데요, $a0과 $a1 레지스터의 값을 더해서 $t0에 담고 있죠. 즉, 스택에 있는 값을 대상으로 연산하는 것이 아닙니다. 저런 식으로 $t0과 $t1 레지스터를 사용한 다음에는 반드시 저 레지스터의 값을 복원해야 합니다. 왜냐하면 해당 procedure가 실행이 완료된 후 돌아갈 코드에서 사용하던 상태로 만들어야 하기 때문입니다.

이로 인해, 그다음 코드에서 lw로 스택에 백업해 둔 값을 다시 가져오고 있습니다.

lw $t0, 0($sp)
lw $t1, 4($sp)

혹시, 이해가 안 된다면 계속 질문해 주세요.
정성태
2021-06-15 01시46분
좀 더 부가적인 설명을 하자면! 다음과 같은 코드가 있다고 가정해 보겠습니다.

j = 3; // $a0에 보관된 값
i = 5; // $a1에 보관된 값
call(i);

그런데, call 프로시저 내부에서 $a0을 사용해야 한다면 어떻게 해야 할까요? 만약 sw로 스택에 보관하지 않고 곧바로 다음과 같은 식으로 사용한다면,

call:
    ...
    add $a0, $a1, $a2
    ...

$a0 레지스터가 덮어써졌기 때문에 call procedure가 반환된 후의,

call(i);
...여기서 j 값은?....

j 값이 깨지게 됩니다. 따라서 저런 경우를 방지하기 위해 procedure에서는 자신이 사용하게 될 레지스터의 값을 백업/복원하는 코드를 반드시 넣게 되는 것입니다.

참고로, Intel CPU의 경우 ebx, ebp, edi, esi 등의 레지스터가 그런 식으로 procedure 간의 호출에서 반드시 호출 전의 상태로 원복해야 한다는 규정이 있습니다. 그래서 그런 레지스터를 가리켜 non-volatile register라고 일컫습니다. 반면 eax, ecx, edx 등의 레지스터는 procedure 간에 값이 당연히 변경될 거라는 가정을 하기 때문에 procedure 내에서 (백업/복원 없이) 자유롭게 사용할 수 있습니다. 그래서 이들을 가리켜 volatile register라고 합니다.

x64 Architecture
; https://docs.microsoft.com/en-us/windows-hardware/drivers/debugger/x64-architecture
정성태

1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5644ㅇㅇ4/6/20224258c# 프로그램을 이용하여 리눅스상에 파일 생성이 가능한가요? [1]
5643유필재4/5/20224133TCP클라이언트 연결 및 통신관련하여 문의드려요 [1]
5642차가워4/4/20224388UdpClient 패킷 수신 문의 [4]
5641장성욱4/4/20224787코어 할당 및 cpu 부하테스트 질문 [7]
5640icoo...4/4/20224312웹가든에서 메모리 동적 업데이트 방법 [1]
5639차가워4/4/20224233c++ 서버 c# 클라이언트 호환 문의 [1]
5638초급4/3/20224540c# sql server 연동 [1]
5637따봉이4/1/20224802Winform Form Load 후 자동 캡쳐관련 [1]파일 다운로드1
5636김철순3/31/20224770WPF에서 Richtext의 View 문의 [5]
5635guest3/30/20224761안정적인 pinning이 가능하네요. [3]파일 다운로드1
5633꿀주세요3/30/20224383선생님 마우스 클릭이벤트 질문이 있습니다. [4]
5632김현수3/30/20224759Remote Desktop으로 접속시 WPF UI 가 다시 그려지는 이벤트를 막을 수 없을까요? [3]
5631김기헌3/24/20224366WPF 컨트롤의 그래픽 처리관련 질문드립니다 [2]파일 다운로드1
5630장성욱3/24/20224178로깅관련 질문입니다. [2]
5629감사합니...3/23/20224418함수에서 예외가 발생하면 try ~ catch처리기를 찾을 때 까지 상위 함수로 계속 올라가나요? [2]
5628홍길동3/23/20225072질문드립니다. [2]파일 다운로드1
5626연준혁3/21/20224325안녕하세요. [3]
5625jaew...3/18/20225270c# 8.0 도서를 구입한 사람입니다. [1]
5624초보자3/17/20224230람다 캡처 관련 문의 [2]
5623한예지 donator3/15/20224636인터프리터 원리가 궁금합니다. [4]
5622김민아3/8/20224545const와 readonly의 명확한 차이가 이게 맞나요? [2]
5621장성욱3/8/20224320c# 로그 관련 질문 [1]
5620김민아3/7/20224356안녕하세요 비관리 객체를 반환하는 메소드 호출 시 궁금한 점이 있어서 질문드립니다 [2]
5619팡팡이3/3/20225718RSA 문의드립니다. [3]
5618김기헌3/2/20224223안녕하세요 생성자 옆에 this 키워드를 붙여 생성자를 여러 개 호출 시 질문드립니다 [2]
5617Edun2/25/20224351ArgumentOutOrRangeException에러 발생 [2]파일 다운로드1
1  2  3  4  5  6  7  8  9  10  11  [12]  13  14  15  ...