180 ไบต์, รหัสเครื่อง (16- บิต x86)
ฉันสังเกตเห็นคำตอบส่วนใหญ่ใช้ builtin encode / decode (ซึ่งฉันเชื่อว่าใช้ได้ดี) แต่ฉันคิดว่าฉันจะทำเควส 16 บิตต่อไปต่อไป
เช่นเดียวกับคนก่อนหน้านี้ก็ทำโดยไม่ต้องคอมไพเลอร์ที่ใช้ส่วนใหญ่HexEditor HTและhexplorer ICY ของ
00000000: eb40 ac20 0000 1a20 9201 1e20 2620 2020 .@. ... ... &
00000010: 2120 c602 3020 6001 3920 5201 0000 7d01 ! ..0 `.9 R...}.
00000020: 0000 0000 1820 1920 1c20 1d20 2220 1320 ..... . . . " .
00000030: 1420 dc02 2221 6101 3a20 5301 0000 7e01 . .."!a.: S...~.
00000040: 7801 89f7 4646 89fa 89d9 4143 4bb4 3fcd x...FF....ACK.?.
00000050: 2185 c074 288a 053c 8073 05e8 1700 ebec !..t(..<.s......
00000060: 3ca0 721a d440 0d80 c050 86c4 e806 0058 <.r..@...P.....X
00000070: e802 00eb d7b4 4088 05b3 01cd 21c3 2c80 ......@.....!.,.
00000080: d0e0 89c3 8b00 89cb 85c0 74c0 3dff 0773 ..........t.=..s
00000090: 08c1 c002 c0e8 02eb cd50 c1e8 0c0c e0e8 .........P......
000000a0: d3ff 5825 ff0f c1c0 02c0 e802 0d80 8050 ..X%...........P
000000b0: 86c4 ebb8 ....
การผ่า
การใช้งานนั้นค่อนข้างตรงไปตรงมาแม้ว่าฉันจะไม่ได้คิดมากว่าจะไหลไปข้างหน้าเลย
ฉันจะสั่งผสมเล็กน้อยเพื่อให้ง่ายต่อการติดตาม ...
0000 eb40 jmp 0x42
ข้ามตารางที่จับคู่ตัวอักษร> = 0x80 <0xa0 ไปยังรหัส Unicode
data db ACh,20h, 00h,00h, 1Ah,20h, ...
รายการที่ไม่ถูกต้องถูกเข้ารหัสเป็น 0 พวกเขาจะไม่ถูกแมปกับสิ่งใด
0075 b440 mov ah, 0x40
0077 8805 mov [di], al
0079 b301 mov bl, 0x1
007b cd21 int 0x21
007d c3 ret
ฟังก์ชั่นตัวช่วยที่ใช้ในการพิมพ์ถ่านในal
จะถูกเรียกว่าไม่กี่ครั้ง
0042 89f7 mov di, si
0044 46 inc si
0045 46 inc si
0046 89fa mov dx, di
0048 89d9 mov cx, bx
004a 41 inc cx
004b 43 inc bx
เตรียมการลงทะเบียน ข้อมูลจะถูกอ่านเป็น 0x100 ให้si
ชี้ไปที่ตารางการแปลข้างต้น
004c 4b dec bx
004d b43f mov ah, 0x3f
004f cd21 int 0x21
0051 85c0 test ax, ax
0053 7428 jz 0x7d
อ่านถ่านจาก stdin ข้ามไปที่ 0x7d ถ้า EOF
Sidenote: นี้เป็นจริง (ที่รู้จักกัน แต่สวยดี) ขนาดเล็กเคล็ดลับ 0x7d มีret
นี้จะทำให้เกิดpop sp
, sp
ที่จุดเริ่มต้นถึงปลายส่วนมี00 00
มีและcs:0
ใน DOS มีCD 20
ซึ่งเป็นสาเหตุของการประยุกต์ใช้เพื่อออก
0055 8a05 mov al, [di]
0057 3c80 cmp al, 0x80
0059 7305 jnc 0x60
005b e81700 call 0x75
005e ebec jmp 0x4c
ถ้า char เป็น <0x80 ให้พิมพ์ออกมาและไปที่จุดเริ่มต้นของลูป (เนื่องจากฟังก์ชันตัวช่วยตั้งค่า BX เป็น 1 - stdout การกระโดดจะไปที่dec bx
)
0060 3ca0 cmp al, 0xa0
0062 721a jc 0x7e
0064 d440 aam 0x40
0066 0d80c0 or ax, c080
0069 50 push ax
006a 86c4 xchg ah, al
006c e80600 call 0x75
006f 58 pop ax
0070 e80200 call 0x75
0073 ebd7 jmp 0x4c
ส่วนนี้เกี่ยวข้องกับ chars> = 0xa0 แยกรหัส ascii เป็น "high" สองบิตและ "low" 6 bits และใช้ utf-8 mask c080 สองไบต์แล้วพิมพ์ทั้งสองอย่าง
007e 2c80 sub al, 0x80
0080 d0e0 shl al, 0x1
0082 89c3 mov bx, ax
0084 8b00 mov ax, [bx+si]
0086 89cb mov bx, cx
0088 85c0 test ax, ax
008a 74c0 jz 0x4c
008c 3dff07 cmp ax, 07ff
008f 7308 jnc 0x99
0091 c1c002 rol ax, 0x2
0094 c0e802 shr al, 0x2
0097 ebcd jmp 0x66
ส่วนนี้เกี่ยวข้องกับ chars> = 0x80 <0xa0 จะพบรหัส utf-8 ที่เหมาะสมในตารางด้านบนหากรหัสเท่ากับ 0 เพียงข้ามไปเริ่มต้นหากต่ำกว่า 0x7ff (ergo: พอดีกับ UTF-8 สองไบต์) เพียงแค่ปรับค่าและใช้รหัสก่อนหน้านี้อีกครั้งที่ 0x166
0099 50 push ax
009a c1e80c shr ax, 0xc
009d 0ce0 or al, e0
009f e8d3ff call 0x75
00a2 58 pop ax
00a3 25ff0f and ax, 0fff
00a6 c1c002 rol ax, 0x2
00a9 c0e802 shr al, 0x2
00ac 0d8080 or ax, 8080
00af 50 push ax
00b0 86c4 xchg ah, al
00b2 ebb8 jmp 0x6c
ส่วนสุดท้ายเกี่ยวข้องกับรหัสที่อยู่เหนือ 0x7FF ปล่อยบิตต่ำ 12 ใช้ 0xE0 (ดูคำอธิบายการเข้ารหัส UTF-8สำหรับการอ้างอิง) และพิมพ์ออกมาปรับ 12 บิตต่ำกว่าและใช้หน้ากาก 8080 และนำส่วนที่แยกออกมาสองครั้งออกมาใหม่ .