aboutsummaryrefslogtreecommitdiff
path: root/files/ko/webassembly/understanding_the_text_format/index.html
blob: 45947c12bab321047e2225d320022ec751ebe9b9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
---
title: Understanding WebAssembly text format
slug: WebAssembly/Understanding_the_text_format
tags:
  - 공유 주소
  - 메모리
  - 웹어셈블리
  - 자바스크립트
  - 테이블
  - 텍스트 포맷
  - 함수
translation_of: WebAssembly/Understanding_the_text_format
---
<div>{{WebAssemblySidebar}}</div>

<p class="summary">사람이 WebAssembly를 읽고 편집 할 수 있게하려면 wasm 이진 형식의 텍스트 표현이 있어야합니다. 이것은 텍스트 편집기, 브라우저 개발자 도구 등에서 노출되도록 고안된 중간 양식입니다.이 장에서는 원시 구문과 텍스트 형식이 나타내는 기본 바이트 코드와 관련하여 Text format이 작동하는 방식과 자바 스크립트에서 wasm을 나타내는 객체 래퍼에 대해 설명합니다.</p>

<div class="note">
<p><strong>참고</strong>: 여기서 다루는 내용은 여러분이 웹어셈블리를 자바스크립트에 바로 불러오는 이전의 방법보다 훨씬 어렵습니다.(<a href="/en-US/docs/WebAssembly/Using_the_JavaScript_API">웹어셈블리를 자바스크립트 API에 사용하기</a> 참고), 하지만 여기서 배우는 내용을 통해 웹어셈블리 모듈을 작성해보면, 자바스크립트 라이브러리의 성능을 향상시킬수 있는 방법을 찾거나, 직접 웹어셈블리 컴파일러를 작성하는데 도움이 될 것입니다.</p>
</div>

<h2 id="S-expressions">S-expressions</h2>

<p>WebAssembly에서 바이너리와 텍스트 사이에 기본적인 코드 교환 방식을 모듈이라 칭합니다. 텍스트 형식에서는, 모듈은 하나의 큰 S-expressions로 표현됩니다. S-expressions는 트리를 텍스트 형식으로 묘사하는 오래되고 쉬운 방법입니다. 모듈을 모듈의 구조와 코드로 표현되는 노드의 트리를 작성하기 위한 수단으로 생각하시면 됩니다. 추상적인 문법을 가진 개발 언어와는 다르게 웹어셈블리의 트리는 단순하면서 일반적으로 많이 사용하는 방식으로 구성되었습니다. </p>

<p> </p>

<p>우선, S- expressions이 어떻게 보이는지 봅시다. 트리의 각 노드는 한 쌍의 괄호<code>( ... )</code> 안에 있습니다. 괄호 안의 첫 번째 레이블은 노드의 유형을 알려주고, 그 후에 속성 또는 하위 노드의 공백으로 구분 된 목록이 있습니다. 즉, WebAssembly S-expression을 의미합니다.</p>

<pre>(module (memory 1) (func))</pre>

<p>모듈이라는 최상위 노드와 2개의 자식 노드를 가진 트리로 구성되었습니다. 그리고 "메모리" 노드에서는 1이라는 속성으로 "함수" 노드를 지칭하였습니다. 이제 이 표현식이 가지는 의미를 간단하게 알아보겠습니다.</p>

<p> </p>

<h3 id="간단한_모듈">간단한 모듈</h3>

<p>가장 간단하고, 작동 가능한 wasm 모듈 작성을 시작합니다.</p>

<pre>(module)</pre>

<p>이 모듈은 전체적으로 비어 있지만 올바르게 작동하는 모듈입니다.</p>

<p>만약 이 모듈을 바이너리로 전환하면,(<a href="/ko/docs/WebAssembly/Text_format_to_wasm">웹어셈블리 텍스트 형식을 wasm으로 변환</a> 참조), 우리는 8바이트짜리 모듈 헤더를 <a href="http://webassembly.org/docs/binary-encoding/#high-level-structure">이진 형식</a>으로 보게 될 것입니다.</p>

<pre>0000000: 0061 736d              ; WASM_BINARY_MAGIC
0000004: 0d00 0000              ; WASM_BINARY_VERSION</pre>

<h3 id="Adding_functionality_to_your_module">Adding functionality to your module</h3>

<p>별로 흥미롭지 않은 모양이군요. 그렇다면 한번 실행 가능한 모듈을 작성해 보도록 하겠습니다.</p>

<p> </p>

<p>webassembly 모듈의 모든 코드는 다음과 같은 의사 코드 구조를 갖는 함수로 구성되어있습니다.</p>

<pre>( func &lt;signature&gt; &lt;locals&gt; &lt;body&gt; )</pre>

<ul>
 <li>명칭(<strong>signature)</strong>은 함수에서 (인자를)받고 (반환 값)반환하는 형식을 정의합니다.</li>
 <li>지역인수(<strong>locals)</strong>는 자바스크립트의 변수 같지만, 명시적으로 형식을 정의합니다.</li>
 <li>본문(<strong>body)</strong>은 저수준 정의를 일렬로 나열한 목록입니다.</li>
</ul>

<p>좀 다르게 보여도 다른 언어의 함수와 비슷합니다.</p>

<p> </p>

<h2 id="Signatures_and_parameters">Signatures and parameters</h2>

<p>Signatures는 반환 형식 선언 목록 뒤에 오는 매개 변수 형식 선언 시퀀스입니다. 여기서 주목할 것은</p>

<ul>
 <li><code>(result)</code>가 없으면 함수는 아무것도 반환하지 않습니다.</li>
 <li>현재 단 하나의 반환 형식을 가질 수 있으나,  <a href="https://webassembly.org/docs/future-features#multiple-return">이후에</a> 여러개를 반환할 수 있게 될 것입니다.</li>
</ul>

<p>각 인자로 wasm은 현재 4가지의 형식을 지원합니다.</p>

<ul>
 <li><code>i32</code>: 32-bit integer</li>
 <li><code>i64</code>: 64-bit integer</li>
 <li><code>f32</code>: 32-bit float</li>
 <li><code>f64</code>: 64-bit float</li>
</ul>

<p> </p>

<p>하나의 인자를 받기 위해 <code>(param i32)</code>라고 작성하고, 반환 값을 받기 위해 <code>(result i32)</code>라고 작성합니다. 아래에 2개의 32비트 정수를 받고 64비트 부동소수를 반환하는 바이너리 함수를 작성하였습니다.</p>

<pre>(func (param i32) (param i32) (result f64) ... )</pre>

<p> </p>

<p>signature 뒤에는 형식을 가진 locals를 나열합니다. <code>(local i32)</code>와 같이 씁니다. parameter는 기본적으로 locals에 속하며, 함수 호출 시 인자에 값을 전달받아 초기화 됩니다.</p>

<h2 id="local와_parameter를_getting_setting_하기">local와 parameter를 getting, setting 하기</h2>

<p>지역인수와 함수인자는 함수 본문에서 <code>get_local</code> 명령문과 <code>set_local</code> 명령문을 통해 가져오거나 설정할 수 있습니다.</p>

<p>The <code>get_local</code>/<code>set_local</code> 명령문은 숫자로 이루어진 요소를 가져오거나 설정합니다. parameter가 선언 순서상 먼저 위치하며, 그다음 locals 순으로 되어 있습니다. </p>

<pre>(func (param i32) (param f32) (local f64)
  get_local 0
  get_local 1
  get_local 2)</pre>

<p>본문 첫 줄에 <code>get_local 0</code> 명령어로 i32 매게변수를 받아내게 되며, <code>get_local 1</code> 명령어로 f32 매게변수를 받게 될 것입니다. 그리고 <code>get_local 2</code> 명렁어로 f64 지역변수를 받을 수 있습니다.</p>

<p>여기서 하나의 문제가 생겼군요. 숫자로 된 순서대로 받자니 좀 혼란스럽고 짜증날 수 있습니다. 그래서 텍스트로 명명된 매게변수, 지역변수, 그리고 다른 요소들을 간편하게 달러문자 (<code>$</code>) 로 시작하여 선언할 수 있습니다.</p>

<p>따라서 위에 작성한 함수 명칭을 아래와 같이 재구성할 수 있습니다.</p>

<pre>(func (param $p1 i32) (param $p2 f32) (local $loc f64) …)</pre>

<p>이렇게 작성하면 <code>get_local 0</code> 대신 <code>get_local $p1</code> 처럼 표현할 수 있습니다. (참고로 여기서 사용된 $인자는 바이너리로 변환 시 다시 숫자로 바뀌게 됩니다.)</p>

<p> </p>

<h2 id="Stack_machines">Stack machines</h2>

<p>우리가 함수 본문을 작성하기 전에 한가지 얘기할 게 더 있습니다. 바로 <strong>스택머신(</strong><strong>stack machines)</strong>이죠. 브라우저가 효율적으로 컴파일할  때, wasm 실행부 안에는 스택 머신에 대한 규칙이 정의되어 있습니다. 간단하게 생각하면 모든 형식을 스택에 넣고, 특정한 <code>i32</code>/<code>i64</code>/<code>f32</code>/<code>f64</code> 값을 스택에서 빼거나 넣는 식입니다.</p>

<p>예를 들면, <code>get_local</code>은 지역변수 값을 스택에 푸시하도록 정의됩니다. 그리고 <code>i32.add</code> 명령어로 두 개의 <code>i32</code> 값을 빼낸 다음(암묵적으로 스택에 넣은 2 개의 값을 가져갑니다.),  이 둘의 합(2^32 형태)을 계산한 다음, 결과로 나온 i32 값을 다시 넣게 됩니다.</p>

<p>함수가 호출되면, 이 함수는 빈 스택으로 시작하여 함수가 실행되는 동안 스택이 서서히 채우고 끝나면 다시 비워 버립니다. 아래 함수를 실행해 봅시다.</p>

<pre>(func (param $p i32)
  get_local $p
  get_local $p
  i32.add)</pre>

<p>여기선 스택이 <code>i32</code> 값 딱 하나만 존재합니다. 그리고 (<code>$p + $p</code>) 식을 <code>i32.add</code> 명령어로 합한 결과가 나오죠. 결국 함수가 가지는 최후의 스택은 반환되는 값 하나 뿐입니다.</p>

<p>웹어셈블리 내 스택 머신의 유효성 보증은 명확합니다. 만약 <code>(result f32)</code> 식을 정의하면, 스택은 반드시 끝에 단 하나의  <code>f32</code> 값만 남아야 합니다. 만약 결과 형식이 정의되어 있지 않다면, 스택은 비어 있어야 하죠.</p>

<p> </p>

<h2 id="첫번째_함수_본문">첫번째 함수 본문</h2>

<p>전에 언급했듯이, 함수의 본문은 단순히  함수가 호출됐을 때 실행할 작업의 리스트입니다. 이전에 다루었던 것을 종합하면 아래와 같이 간단한 함수를 포함하는 모듈을 선언할 수 있습니다.</p>

<pre>(module
  (func (param $lhs i32) (param $rhs i32) (result i32)
    get_local $lhs
    get_local $rhs
    i32.add
  )
)</pre>

<p>이 함수는 2개의 매개변수를 갖고, 둘을 더하고, 그 결과를 반환합니다.</p>

<p>함수의 본문에 넣을 수 있는 것들은 훨씬 많지만 지금은 간단하게 시작할겁니다. 당신은 앞으로 더 많은 예제들을 보게 될 것입니다. 사용 가능한 opcodes의 모든 목록을 보고싶다면 <a href="http://webassembly.org/docs/semantics/">webassembly.org 시맨틱 문서</a>에서 확인하세요.</p>

<p> </p>

<h3 id="함수_호출하기">함수 호출하기</h3>

<p>함수는 혼자서 동작하지 않습니다. 호출해야합니다. ES2015 모듈에서와 마찬가지로 wasm 함수는 모듈 내부의 <code>export</code> 명령문에 의해 명시적으로 내보내야합니다.</p>

<p>locals와 마찬가지로 함수는 기본적으로 인덱스로 식별되지만 편의상 이름을 지정할 수 있습니다. 먼저 <code>func</code> 키워드 바로 뒤에 달러 기호가 붙은 이름을 추가합니다.</p>

<pre>(func $add … )</pre>

<p>이제 내보내기 선언을 추가해야합니다. 다음과 같습니다.</p>

<pre>(export "add" (func $add))</pre>

<p>여기서 <code>add</code>는 자바 스크립트에서 함수가 식별되는 이름이며 <code>$add</code>는 모듈 내에서 어떤 WebAssembly 함수가 내보내지는지를 선택합니다.</p>

<p>그래서 우리의 마지막 모듈은 (지금은) 다음과 같습니다.</p>

<pre>(module
  (func $add (param $lhs i32) (param $rhs i32) (result i32)
    get_local $lhs
    get_local $rhs
    i32.add)
  (export "add" (func $add))
)</pre>

<p>예제를 따라하려면 위의 모듈을 <code>add.wat</code>라는 파일에 저장 한 다음 wabt를 사용하여 <code>add.wasm</code>이라는 이진 파일로 변환하십시오 (자세한 내용은 <a href="/en-US/docs/WebAssembly/Text_format_to_wasm">Converting WebAssembly text format to wasm</a> 참조).</p>

<p>다음으로 바이너리를 <code>addCode</code> (<a href="/en-US/docs/WebAssembly/Fetching_WebAssembly_bytecode">Fetching WebAssembly Bytecode</a>에서 설명한대로)라는 형식화 된 배열에로드하고, 컴파일 및 인스턴스화 한 다음 자바 스크립트에서 <code>add</code> 함수를 실행합니다. (이제 <code>add()</code>는 인스턴스의 <code><a href="/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/Instance/exports">exports</a></code> 속성에서 찾을 수 있습니다)</p>

<pre><code>WebAssembly.instantiateStreaming(fetch('add.wasm'))
.then(obj =&gt; {
   console.log(obj.instance.exports.add(1, 2));  // "3"
});</code></pre>

<div class="note">
<p><strong>참고:</strong> 이 예제는 GitHub에서 <a href="https://github.com/mdn/webassembly-examples/blob/master/understanding-text-format/add.html">add.html</a>로 찾을 수 있습니다 (<a href="https://mdn.github.io/webassembly-examples/understanding-text-format/add.html">see it live also</a>). 인스턴스 함수에 대한 자세한 내용은 {{jsxref ( "WebAssembly.instantiate ()")}}를 참조하십시오.</p>
</div>

<h2 id="기본_사항_둘러보기">기본 사항 둘러보기</h2>

<p>이제는 기본 사항을 다뤘습니다. 더 많은 고급 기능을 살펴 보겠습니다.</p>

<h3 id="같은_모듈_내에_있는_다른_함수_호출하기">같은 모듈 내에 있는 다른 함수 호출하기</h3>

<p><code>call</code> 명령어는 인덱스 또는 이름이 지정된 단일 함수를 호출합니다. 예를 들어, 다음 모듈에는 두 개의 함수가 있습니다. 하나는 값 42를 반환하고, 다른 하나는 첫 번째 플러스 1을 호출 한 결과를 반환합니다.</p>

<pre>(module
  (func $getAnswer (result i32)
    i32.const 42)
  (func (export "getAnswerPlus1") (result i32)
    call $getAnswer
    i32.const 1
    i32.add))</pre>

<div class="note">
<p><strong>참고:</strong> <code>i32.const</code>는 단지 32 비트 정수를 정의하고 그것을 스택에 푸시합니다. 사용할 수있는 다른 유형의 <code>i32</code>를 바꿀 수 있으며 원하는 값으로 const 값을 변경할 수 있습니다 (여기서는 값을 <code>42</code>로 설정했습니다).</p>
</div>

<p>In this example you’ll notice an <code>(export "getAnswerPlus1")</code> section, declared just after the <code>func</code> statement in the second function — this is a shorthand way of declaring that we want to export this function, and defining the name we want to export it as.</p>

<p>이는 함수 밖에서, 예전과 같은 방식으로 모듈의 다른 곳에서 별도의 function 문을 포함하는 것과 기능적으로 동일합니다. 예 :</p>

<pre>(export "getAnswerPlus1" (func $functionName))</pre>

<p>위의 모듈을 호출하는 JavaScript 코드는 다음과 같습니다.</p>

<pre><code>WebAssembly.instantiateStreaming(fetch('call.wasm'))
.then(obj =&gt; {
   console.log(obj.instance.exports.getAnswerPlus1());  // "43"
});</code></pre>

<div class="note">
<p><strong>참고:</strong> 이 예제는 GitHub에서 <a href="https://github.com/mdn/webassembly-examples/blob/master/understanding-text-format/call.html">call.html</a>로 찾을 수 있습니다 (<a href="https://mdn.github.io/webassembly-examples/understanding-text-format/call.html">see it live also</a>). </p>
</div>

<h3 id="자바스크립트에_함수_가져오기">자바스크립트에 함수 가져오기</h3>

<p>우리는 이미 JavaScript에서 WebAssembly 함수를 호출하는 것을 보았습니다. 그러나 JavaScript 함수를 호출하는 WebAssembly는 어떻습니까? WebAssembly에는 실제로 JavaScript에 대한 기본 지식이 없지만 JavaScript 또는 Wasm 함수를 사용할 수있는 함수를 가져 오는 일반적인 방법이 있습니다. 예제를 살펴 보겠습니다.</p>

<pre>(module
  (import "console" "log" (func $log (param i32)))
  (func (export "logIt")
    i32.const 13
    call $log))</pre>

<p>WebAssembly에는 2 단계 네임 스페이스가 있으므로 여기서 import 문은 콘솔<code>console</code>모듈에서 <code>log</code> 함수를 가져 오기를 요청한다는 의미입니다. 또한 앞서 소개 한 <code>call</code> 명령어를 사용하여 내 보낸 <code>logIt</code> 함수가 가져온 함수를 호출하는지 확인할 수 있습니다.</p>

<p>가져온 함수는 일반적인 함수와 같습니다. WebAssembly 유효성 검사에서 정적으로 서명하는 시그니처가 있으며, 인덱스가 주어지며 이름을 지정하고 호출 할 수 있습니다.</p>

<p>JavaScript 함수에는 서명 개념이 없으므로 임포트의 선언 된 서명에 관계없이 모든 JavaScript 함수를 전달할 수 있습니다. 모듈이 가져 오기를 선언하면 {{jsxref("WebAssembly.instantiate()")}}의 호출자는 해당 속성이있는 가져 오기 객체를 전달해야합니다.</p>

<p>위의 경우 <code>importObject.console.log</code>가 JavaScript 함수 인 객체 (<code>importObject</code>라고 부름)가 필요합니다.</p>

<p>이것은 다음과 같습니다.</p>

<pre><code>var importObject = {
  console: {
    log: function(arg) {
      console.log(arg);
    }
  }
};

WebAssembly.instantiateStreaming(fetch('logger.wasm'), importObject)
.then(obj =&gt; {
  obj.instance.exports.logIt();
});</code></pre>

<div class="note">
<p><strong>Note</strong>: 이 예제는 GitHub에서 <a href="https://github.com/mdn/webassembly-examples/blob/master/understanding-text-format/logger.html">logger.html</a>로 찾을 수 있습니다 (라이브(<a href="https://mdn.github.io/webassembly-examples/understanding-text-format/logger.html">see it live also</a>)도 참조하십시오).</p>
</div>

<p> </p>

<h3 id="Declaring_globals_in_WebAssembly">Declaring globals in WebAssembly</h3>

<p>WebAssembly는 하나 이상의 {{jsxref("WebAssembly.Module")}} 인스턴스 전체에서 JavaScript 및 가져오기 / 내보내기가 가능한 전역 변수 인스턴스를 생성 할 수 있습니다. 이는 여러 모듈을 동적으로 연결할 수 있으므로 매우 유용합니다.</p>

<p>WebAssembly 텍스트 형식에서는 다음과 비슷합니다 (GitHub의 <a href="https://github.com/mdn/webassembly-examples/blob/master/js-api-examples/global.wat">global.wat</a> 참조, 라이브 JavaScript 예제는 <a href="https://mdn.github.io/webassembly-examples/js-api-examples/global.html">global.html</a> 참조).</p>

<pre><code>(module
   (global $g (import "js" "global") (mut i32))
   (func (export "getGlobal") (result i32)
        (get_global $g))
   (func (export "incGlobal")
        (set_global $g
            (i32.add (get_global $g) (i32.const 1))))
)</code></pre>

<p>이것은 <code>global</code> 키워드를 사용하여 전역 값을 지정한다는 점을 제외하고는 이전에 보았던 것과 비슷하게 보이며 변수 mutable을 원하면 값의 데이터 유형과 함께 <code>mut</code> 키워드를 지정합니다.</p>

<p>JavaScript를 사용하여 동일한 값을 만들려면 {{jsxref("WebAssembly.Global()")}} 생성자를 사용합니다.</p>

<pre><code>const global = new WebAssembly.Global({value:'i32', mutable:true}, 0);</code>
</pre>

<p> </p>

<h3 id="WebAssembly_Memory">WebAssembly Memory</h3>

<p> </p>

<p>위의 예로든 로깅 기능은 아주 끔찍합니다. 단 하나의 정수만 출력합니다!(이 정수가 뭔지 어떻게 알겠어요 그죠?) 텍스트 문자열을 기록하려면 어떻게 해야할까요. 문자열 및 기타 복잡한 데이터 유형을 처리하기 위해 WebAssembly는 메모리를 제공합니다. 메모리는 사용하면서 크기를 키울 수 있는 대량의 바이트 배열 입니다. WebAssembly에는 <a href="http://webassembly.org/docs/semantics/#linear-memory">linear memory</a>에서 읽고 쓰는 데 필요한 <code>i32.load</code><code>i32.store</code>와 같은 지침이 들어 있습니다.</p>

<p>JavaScript의 관점에서 볼 때 크기가 조정 가능한 큰 {{domxref("ArrayBuffer")}} 안에 메모리가 모두있는 것처럼 보입니다. 이는 말 그대로 asm.js와 함께 사용해야함을 의미합니다. (크기 조정할 수 없다는 점을 제외하고요. asm.js <a href="http://asmjs.org/spec/latest/#programming-model">Programming model</a>을 참고 하세요.).</p>

<p>따라서 문자열은 이 선형 메모리 내부의 있는 sequence of bytes라고 할 수 있습니다. 우리가 적절한 바이트 문자열을 메모리에 썼다고 가정 해 보고 어떻게 그 문자열을 JavaScript로 전달하는지 보겠습니다.</p>

<p> </p>

<p>핵심은 자바 스크립트가 {{jsxref("WebAssembly.Memory()")}} 인터페이스를 통해 WebAssembly 선형 메모리 인스턴스를 생성하고 연관된 인스턴스를 사용하여 기존 메모리 인스턴스에 액세스 할 수 있다는 것입니다 (현재 모듈 인스턴스 당 하나만 가질 수 있음). 메모리 인스턴스에는 <code><a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/Memory/buffer">buffer</a></code> getter가 있습니다.이 buffer getter는 전체 선형 메모리를 가리키는 <code>ArrayBuffer</code>를 반환합니다.</p>

<p> </p>

<p>예를 들어 JavaScript의 <code><a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/Memory/grow">Memory.grow()</a></code> 메소드를 통해 메모리 인스턴스를 늘릴 수도 있습니다. grow가 발생하면 <code>ArrayBuffer</code>s는 크기를 변경할 수 없기 때문에 현재의 <code>ArrayBuffer</code>가 분리되고 더 큰 새 메모리를 가리 키도록 새 <code>ArrayBuffer</code>가 만들어집니다. 즉, JavaScript에 문자열을 전달하기 위해 수행해야하는 모든 작업은 길이를 나타내는 방법과 함께 선형 메모리에있는 문자열의 오프셋을 전달하는 것입니다.</p>

<p> </p>

<p>문자열 자체의 길이를 인코딩하는 방법에는 여러 가지가 있습니다 (예 : C 문자열). 여기서 간단히하기 위해 offset과 length를 매개 변수로 전달합니다.</p>

<pre><code>(import "console" "log" (func $log (param i32) (param i32)))</code>
</pre>

<p> </p>

<p>자바 스크립트 측면에서 우리는 <a href="https://developer.mozilla.org/en-US/docs/Web/API/TextDecoder">TextDecoder API</a>를 사용하여 바이트를 자바 스크립트 문자열로 쉽게 디코딩 할 수 있습니다. (여기서는 <code>utf8</code>을 지정하지만 다른 많은 인코딩이 지원됩니다.)</p>

<pre><code>function consoleLogString(offset, length) {
  var bytes = new Uint8Array(memory.buffer, offset, length);
  var string = new TextDecoder('utf8').decode(bytes);
  console.log(string);
}</code>
</pre>

<p> </p>

<p>이제 남은 부분은 <code>consoleLogString</code>이 WebAssembly <code>memory</code>에 액세스하는 부분입니다. WebAssembly는 JavaScript로 <code><a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/Memory">Memory</a></code> 객체를 만들고 WebAssembly 모듈에서 메모리를 가져 오거나 WebAssembly 모듈에서 메모리를 만들어 JavaScript로 내보낼 수 있는 유연성을 제공합니다.</p>

<p> </p>

<p> </p>

<p>간단히하기 위해 JavaScript로 작성한 다음 WebAssembly로 가져와 봅시다. 우리의 <code>import</code>statement는 다음과 같이 작성됩니다 :</p>

<pre><code>(import "js" "mem" (memory 1))</code></pre>

<p><code>1</code>은 가져온 메모리에 최소 1 페이지의 메모리가 있어야 함을 나타냅니다 (WebAssembly에서 페이지를 64KB로 정의 함).</p>

<p>문자열 "Hi"를 인쇄하는 완전한 모듈을 보겠습니다. 일반적인 컴파일 된 C 프로그램에서는 함수를 호출하여 문자열에 대한 메모리를 할당하지만 여기에 자체 어셈블리를 작성하고 전체 선형 메모리를 소유하기 때문에 <code>data</code> 섹션을 사용하여 전역 메모리에 문자열 내용을 쓸 수 있습니다. 데이터 섹션은 바이트의 문자열이 인스턴스화 시간에 주어진 오프셋에 쓰여질 수 있도록 하며 원시 실행 가능 형식의 <code>.data</code>sections와 유사합니다.</p>

<p>최종 wasm 모듈을 확인하겠습니다.</p>

<pre><code>(module
  (import "console" "log" (func $log (param i32 i32)))
  (import "js" "mem" (memory 1))
  (data (i32.const 0) "Hi")
  (func (export "writeHi")
    i32.const 0  ;; pass offset 0 to log
    i32.const 2  ;; pass length 2 to log
    call $log))</code></pre>

<div class="note">
<p><strong>Note</strong>: 위와 같이 WebAssembly 파일의 주석을 허용하는 두 개의 세미콜론 구문 (<code>;;</code>)을 확인하십시오.</p>
</div>

<p>자바 스크립트에서 우리는 1 페이지 메모리를 만들고 그것을 전달할 수 있습니다. 결과적으로 "Hi"가 콘솔에 출력됩니다 :</p>

<pre><code>var memory = new WebAssembly.Memory({initial:1});

var importObject = { console: { log: consoleLogString }, js: { mem: memory } };

WebAssembly.instantiateStreaming(fetch('logger2.wasm'), importObject)
.then(obj =&gt; {
  obj.instance.exports.writeHi();
});</code></pre>

<div class="note">
<p><strong>Note</strong>: 깃허브에서 소스 전체버전을 확인할 수 있습니다. <a href="https://github.com/mdn/webassembly-examples/blob/master/understanding-text-format/logger2.html">logger2.html</a> (<a href="https://mdn.github.io/webassembly-examples/understanding-text-format/logger2.html">also see it live</a>).</p>
</div>

<h3 id="WebAssembly_tables">WebAssembly tables</h3>

<p>WebAssembly 텍스트 형식 둘러보기를 마무리하려면 WebAssembly에서 가장 복잡하고 어려운 부분 인 <strong>tables</strong>을 살펴 보겠습니다. 테이블은 기본적으로 WebAssembly 코드의 인덱스를 통해 액세스 할 수 있는 크기 조정이 가능한 참조 배열입니다.</p>

<p>왜 테이블이 필요한지를 알기 위해서는 앞서 보았던 <code>call</code> 명령어 ({{anch("Calling functions from other functions in the same module")}} 참고)가 필요합니다. 이 명령어는 정적 함수 인덱스를 취하며 따라서 오직 하나의 함수를 호출합니다. 그러나 호출 수신자가 런타임 값인 경우에는 어떻게 해야할까요?</p>

<ul>
 <li>자바 스크립트에서 항상 볼 수 있습니다. 함수는 최상위 클래스 입니다.</li>
 <li>C/C++에서, function pointers와 함께 볼 수 있습니다.</li>
 <li>C++에서, virtual functions와 함께 볼 수 있습니다.</li>
</ul>

<p>WebAssembly에는 이를 구현하기 위해 일종의 호출 명령어가 필요했기 때문에 동적 함수 피연산자를 사용하는 <code>call_indirect</code>를 구현했습니다. 문제는 WebAssembly에서 피연산자를 지정해야하는 유일한 유형은 (현재) <code>i32</code>/<code>i64</code>/<code>f32</code>/<code>f64</code>라는 것입니다.</p>

<p>WebAssembly는 <code>anyfunc</code> 유형을 추가 할 수 있습니다 (유형이 모든 서명의 기능을 보유 할 수 있기 때문에 "any"가 붙어있습니다). 그러나 안타깝게도 이 <code>anyfunc</code> 유형은 보안상의 이유로 선형 메모리에 저장할 수 없습니다. 선형 메모리는 저장된 값의 원시 내용을 바이트로 표시하므로 wasm 내용이 임의로 웹에서 허용 하면 안되는 원시 함수 주소를 노출하거나 손상시킬 수 있습니다.</p>

<p> </p>

<p>해결책은 테이블에 함수 참조를 저장하고 대신 테이블 인덱스를 전달하는 것입니다.이 인덱스는 i32 값입니다. <code>call_indirect</code>의 피연산자는 단순히 i32 인덱스 값일 수 있습니다.</p>

<p> </p>

<h4 id="Defining_a_table_in_wasm">Defining a table in wasm</h4>

<p>그러면 우리 테이블에 함수를 어떻게 배치할까요? <code>data</code> 섹션을 사용하여 선형 메모리 영역을 바이트로 초기화하는 것처럼 <code>elem</code> 섹션을 사용하여 함수가있는 테이블 영역을 초기화 할 수 있습니다.</p>

<pre><code>(module
  (table 2 anyfunc)
  (elem (i32.const 0) $f1 $f2)
  (func $f1 (result i32)
    i32.const 42)
  (func $f2 (result i32)
    i32.const 13)
  ...
)</code></pre>

<ul>
 <li><code>(table 2 anyfunc)</code>에서 2는 테이블의 초기 크기 (즉, 두 개의 참조를 저장함)이고 <code>anyfunc</code>는 이러한 참조의 요소 유형이 "any signature가 있는 함수"로 선언합니다. WebAssembly의 현재 반복에서는 이것이 유일하게 허용되는 요소 유형이지만, 앞으로 더 많은 요소 유형이 추가 될 것입니다.</li>
 <li>함수 (<code>func</code>) 섹션은 다른 선언 된 wasm 함수와 같습니다. 이것들은 테이블에서 참조 할 함수입니다 (예를 들어, 각각은 상수 값을 반환합니다). 섹션이 선언 된 순서는 중요하지 않습니다. 여기서는 함수를 어디에서나 선언 할 수 있으며 여전히 <code>elem</code> 섹션에서 함수를 참조 할 수 있습니다.</li>
 <li><code>elem</code> 섹션은 모듈의 모든 함수 서브 세트를 순서에 상관없이 나열하며 중복을 허용 합니다. 이것은 참조되는 순서대로 테이블에서 참조 할 목록입니다.</li>
 <li><code>elem</code> 섹션 내의 <code>(i32.const 0)</code> 값은 오프셋입니다. 이것은 섹션의 시작 부분에서 선언해야하며, 테이블 함수 참조가 채워지기 시작하는 인덱스를 지정합니다. 여기서 0과 2의 크기를 지정 했으므로 (위 참조) 인덱스 0과 1에서 두 개의 참조를 채울 수 있습니다. 오프셋 1에서 참조를 쓰고 싶다면 <code>(i32.const 1)</code>이고 테이블 크기는 3이어야합니다.</li>
</ul>

<p> </p>

<div class="note">
<p><strong>Note</strong>: 초기화되지 않은 요소에는 기본 throw-on-call 값이 제공됩니다.</p>
</div>

<p>자바 스크립트에서 이와 같은 테이블 인스턴스를 생성하는 호출은 다음과 같습니다.</p>

<pre><code>function() {
  // table section
  var tbl = new WebAssembly.Table({initial:2, element:"anyfunc"});

  // function sections:
  var f1 = function() { … }
  var f2 = function() { … }

  // elem section
  tbl.set(0, f1);
  tbl.set(1, f2);
};</code></pre>

<p> </p>

<h4 id="Using_the_table">Using the table</h4>

<p>우리가 사용 할 테이블을 정의했습니다. 다음 코드 섹션을 통해 테이블을 사용 할 수 있습니다.</p>

<pre><code>(type $return_i32 (func (result i32))) ;; if this was f32, type checking would fail
(func (export "callByIndex") (param $i i32) (result i32)
  get_local $i
  call_indirect (type $return_i32))</code></pre>

<ul>
 <li><code>(type $return_i32 (func (param i32)))</code> 블록은 참조 이름과 함께 유형을 지정합니다. 이 유형은 나중에 테이블 함수 참조 호출의 유형 점검을 수행 할 때 사용됩니다. 여기서 우리는 참조가 <code>i32</code>를 결과로 반환하는 함수라고 코드를 작성했습니다.</li>
 <li>다음으로, <code>callByIndex</code>라는 이름으로 익스포트 될 함수를 정의합니다. 이것은 하나의 <code>i32</code>를 매개 변수로 사용하며, 인수 이름 <code>$i</code>가 주어집니다.</li>
 <li>Inside the function, we add one value to the stack — whatever value is passed in as the parameter <code>$i</code>. 매개 변수 <code>$i</code>로 전달되는 값이 무엇이든간에 함수 내에서 스택에 하나의 값을 추가합니다.</li>
 <li>마지막으로, <code>call_indirect</code>를 사용하여 테이블에서 함수를 호출합니다. 스택에서 <code>$i</code>의 값을 암시 적으로 팝합니다. 이것의 최종 결과는 <code>callByIndex</code> 함수가 테이블에서 <code>$i</code>'th 함수를 호출한다는 것입니다.</li>
</ul>

<p> </p>

<p>다음과 같이 <code>call_indirect</code> 매개 변수를 명령 호출 중에 명시 적으로 선언 할 수도 있습니다.</p>

<pre><code>(call_indirect (type $return_i32) (get_local $i))</code></pre>

<p>자바 스크립트와 같이 더 높은 수준의 표현력있는 언어에서는 함수가 포함 된 배열 (또는 더 많은 가능성이있는 객체)을 사용하여 동일한 작업을 수행한다고 상상할 수 있습니다. 의사 코드는 <code>tbl[i]()</code>와 유사합니다.</p>

<p> </p>

<p> </p>

<p>다시 typechecking을 보면 WebAssembly가 typechecked되고 <code>anyfunc</code>가 "모든 함수 서명"을 의미하기 때문에 callee에서 호출 수신자의 추정 서명을 제공해야하므로 <code>$return_i32</code> 유형이 포함되어 <code>i32</code>를 반환하는 함수가 예상됩니다. 호출 수신자와 일치하는 서명이 없으면 (대신 <code>f32</code>가 반환 됨) {{jsxref("WebAssembly.RuntimeError")}}가 발생합니다.</p>

<p><code>call_indirect</code>와 호출하는 테이블을 연결하는 것은 무엇입니까? 대답은 모듈 인스턴스 당 하나의 테이블 만 허용된다는 것입니다. <code>call_indirect</code>가 암시 적으로 호출하는 것입니다. 향후, 여러 테이블이 허용 될 때, 우리는 또한 어떤 종류의 테이블 식별자를 지정할 필요가있습니다</p>

<p> </p>

<p> </p>

<pre><code>call_indirect $my_spicy_table (type $i32_to_void)</code></pre>

<p>전체 모듈은 모두 이와 같이 보이며 <a href="https://github.com/mdn/webassembly-examples/blob/master/understanding-text-format/wasm-table.wat">wasm-table.wat</a> 예제 파일에서 찾을 수 있습니다.</p>

<p> </p>

<pre><code>(module
  (table 2 anyfunc)
  (func $f1 (result i32)
    i32.const 42)
  (func $f2 (result i32)
    i32.const 13)
  (elem (i32.const 0) $f1 $f2)
  (type $return_i32 (func (result i32)))
  (func (export "callByIndex") (param $i i32) (result i32)
    get_local $i
    call_indirect (type $return_i32))
)</code></pre>

<p> </p>

<p>다음 자바 스크립트를 사용하여 웹 페이지에로드합니다.</p>

<pre><code>WebAssembly.instantiateStreaming(fetch('wasm-table.wasm'))
.then(obj =&gt; {
  console.log(obj.instance.exports.callByIndex(0)); // returns 42
  console.log(obj.instance.exports.callByIndex(1)); // returns 13
  console.log(obj.instance.exports.callByIndex(2)); // returns an error, because there is no index position 2 in the table
});</code></pre>

<div class="note">
<p><strong>Note</strong>: <a href="https://github.com/mdn/webassembly-examples/blob/master/understanding-text-format/wasm-table.html">wasm-table.html</a>에서 이 예제를 확인할 수 있습니다. (<a href="https://mdn.github.io/webassembly-examples/understanding-text-format/wasm-table.html">see it live also</a>).</p>
</div>

<div class="note">
<p><strong>Note</strong>: Memory와 마찬가지로 테이블은 자바 스크립트 (<code><a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/Table">WebAssembly.Table()</a></code> 참고)와 다른 wasm 모듈로 가져 오거나 다른 wasm 모듈에서 가져올 수도있다.</p>
</div>

<h3 id="Mutating_tables_and_dynamic_linking">Mutating tables and dynamic linking</h3>

<p> </p>

<p>JavaScript는 함수 참조에 대한 모든 액세스 권한을 갖기 때문에 Grow (), get () 및 set () 메서드를 통해 JavaScript에서 Table 객체를 변형 할 수 있습니다. WebAssembly에서 참조 유형을 가져 오면 WebAssembly 코드는 get_elem / set_elem 명령어로 테이블 자체를 변형 할 수 있습니다.</p>

<p>테이블은 변경 가능하기 때문에 정교한 로드 시간 및 런타임 <a href="http://webassembly.org/docs/dynamic-linking">dynamic linking schemes</a>를 구현하는 데 사용할 수 있습니다. 프로그램이 동적으로 링크되면 여러 인스턴스가 동일한 메모리 및 테이블을 공유합니다. 이것은 여러 컴파일 된 <code>.dll</code>이 단일 프로세스의 주소 공간을 공유하는 기본 응용 프로그램과 대칭입니다.</p>

<p> </p>

<p>이 작업을 보려면 Memory 객체와 Table 객체가 포함 된 단일 가져 오기 객체를 만들고 동일한 가져 오기 객체를 여러 <code><a href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/WebAssembly/instantiate">instantiate()</a></code> 호출에 전달합니다.</p>

<p><code>.wat</code> 예제는 다음과 같습니다.</p>

<p><code>shared0.wat</code>:</p>

<pre><code>(module
  (import "js" "memory" (memory 1))
  (import "js" "table" (table 1 anyfunc))
  (elem (i32.const 0) $shared0func)
  (func $shared0func (result i32)
   i32.const 0
   i32.load)
)</code></pre>

<p><code>shared1.wat</code>:</p>

<pre><code>(module
  (import "js" "memory" (memory 1))
  (import "js" "table" (table 1 anyfunc))
  (type $void_to_i32 (func (result i32)))
  (func (export "doIt") (result i32)
   i32.const 0
   i32.const 42
   i32.store  ;; store 42 at address 0
   i32.const 0
   call_indirect (type $void_to_i32))
)</code></pre>

<p>작업은 다음과 같습니다.</p>

<ol>
 <li><code>shared0func</code> 함수는 <code>shared0.wat</code>에 정의되어 있으며 가져온 테이블에 저장됩니다.</li>
 <li>이 함수는 값 <code>0</code>을 포함하는 상수를 작성한 다음 <code>i32.load</code> 명령을 사용하여 제공된 메모리 색인에 포함 된 값을로드합니다. 제공된 인덱스는 <code>0</code>입니다 - 다시 암시 적으로 스택에서 이전 값을 팝합니다. 따라서 <code>shared0func</code>는 메모리 인덱스 <code>0</code>에 저장된 값을로드하고 반환합니다.</li>
 <li><code>shared1.wat</code>에서 <code>doIt</code>이라는 함수를 내 보냅니다.이 함수는 <code>0</code><code>42</code> 값을 포함하는 두 개의 상수를 만든 다음 <code>i32.store</code>를 호출하여 제공된 값을 가져온 메모리의 제공된 색인에 저장합니다. 다시 말하지만, 스택에서이 값을 암시 적으로 팝하므로 메모리 인덱스 <code>0</code>에 값 <code>42</code>를 저장합니다.</li>
 <li>함수의 마지막 부분에서 값 0으로 상수를 만든 다음이 인덱스 0에서 <code>shared0func</code>라는 함수를 호출하고 <code>shared0.wat</code><code>elem</code> 블록에 먼저 저장합니다.</li>
 <li><code>shared0func</code>가 호출되면 <code>shared1.wat</code><code>i32.store</code> 명령을 사용하여 메모리에 저장된 <code>42</code>를 로드합니다.</li>
</ol>

<div class="note">
<p><strong>Note</strong>: 위의 표현식은 암시 적으로 스택의 값을 다시 표시하지만 명령 호출 내에서 명시 적으로 이를 선언 할 수 있습니다. 예를 들면 다음과 같습니다.:</p>

<pre><code>(i32.store (i32.const 0) (i32.const 42))
(call_indirect (type $void_to_i32) (i32.const 0))</code>
</pre>
</div>

<p>어셈블리로 변환 한 후 다음 코드를 통해 JavaScript에서 <code>shared0.wasm</code><code>shared1.wasm</code>을 사용합니다.</p>

<pre><code>var importObj = {
  js: {
    memory : new WebAssembly.Memory({ initial: 1 }),
    table : new WebAssembly.Table({ initial: 1, element: "anyfunc" })
  }
};

Promise.all([
  WebAssembly.instantiateStreaming(fetch('shared0.wasm'), importObj),
  WebAssembly.instantiateStreaming(fetch('shared1.wasm'), importObj)
]).then(function(results) {
  console.log(results[1].instance.exports.doIt());  // prints 42
});</code>
</pre>

<p>컴파일되는 각 모듈은 동일한 메모리 및 테이블 객체를 가져와 동일한 선형 메모리 및 테이블 "주소 공간"을 공유 할 수 있습니다.</p>

<div class="note">
<p><strong>Note</strong>: <a href="https://github.com/mdn/webassembly-examples/blob/master/understanding-text-format/shared-address-space.html">shared-address-space.html</a>에서 예제를 확인할 수 있습니다.(<a href="https://mdn.github.io/webassembly-examples/understanding-text-format/shared-address-space.html">see it live also</a>).</p>
</div>

<h2 id="Summary">Summary</h2>

<p>WebAssembly 텍스트 형식의 주요 구성 요소에 대한 높은 수준의 둘러보기와 WebAssembly JS API에 반영되는 방법을 마무리합니다.</p>

<h2 id="See_also">See also</h2>

<ul>
 <li>포함되지 않은 주요 내용은 함수 본문에서 발생할 수있는 모든 명령의 포괄적 인 목록입니다. 각 명령어의 처리 방법은 <a href="http://webassembly.org/docs/semantics">WebAssembly semantics</a>을 참조하십시오.</li>
 <li>spec 해석기로 구현된 <a href="https://github.com/WebAssembly/spec/blob/master/interpreter/README.md#s-expression-syntax">grammar of the text format</a>도 참조하십시오.</li>
</ul>

<p> </p>