x86-64 मशीन कोड फ़ंक्शन, 53 48 बाइट्स
चैंज:
- विशेष मामले
jz
को संभालने के लिए 64-बिट शिफ्ट का उपयोग करने के बजाय शिफ्ट पर -2 >>(32-0)
।
- एएल के बजाय जेडएफ में -3 वापसी, ए के लिए 3 बाइट्स की बचत
setnz al
।
( इस पर आधारित डैनियल शेप्लेर के 32-बिट मशीन कोड उत्तर भी देखें , जो तब हमारे पास मौजूद कुछ अन्य विचारों का उपयोग करने के लिए विकसित हुआ था। मैं इस उत्तर के निचले भाग में मेरा नवीनतम संस्करण शामिल कर रहा हूं।)
सबनेट में ZF = 0 होस्ट के लिए नहीं , सबनेट में ZF = 1 के लिए लौटाता है , तो आप परिणाम के साथ शाखा कर सकते हैंje host_matches_subnet
X86-64 सिस्टम V कॉलिंग कन्वेंशन के साथ कॉल करने योग्य जैसे
bool not_in_subnet(int dummy_rdi, const char *input_rsi);
कि आप में जोड़ते हैंsetnz al
।
इनपुट स्ट्रिंग में होस्ट और नेटवर्क दोनों शामिल होते हैं, जो बिल्कुल 1-अंकीय वर्ण द्वारा अलग किए जाते हैं। CIDR चौड़ाई के अंत के बाद वाली मेमोरी में पेज के अंत से पहले कम से कम 3 गैर-अंक बाइट्स होना चाहिए। (ज्यादातर मामलों में एक समस्या नहीं होनी चाहिए, जैसे कि cmdline arg के लिए।) डैनियल के 32-बिट संस्करण में यह सीमा नहीं है।
हम एक ही डॉटेड-क्वाड पार्स लूप को 3 बार चलाते हैं, दो आईपीवी 4 पते प्राप्त कर रहे हैं, और प्राप्त कर रहे हैं /mask
एक डॉर्ड के उच्च बाइट में पूर्णांक रूप में । (यही कारण है कि बाद में पढ़ने योग्य मेमोरी होनी चाहिए /mask
, लेकिन एएससीआईआई अंक होने पर कोई फर्क नहीं पड़ता।)
हम (host ^ subnet) >> (32-mask)
मेजबान बिट्स (बेमेल की अनुमति देने वाले) को बाहर करने के लिए करते हैं, जिससे सबनेट और होस्ट के बीच का अंतर ही रह जाता है। उस /0
विशेष मामले को हल करने के लिए जहां हमें 32 से शिफ्ट करने की आवश्यकता है, हम काउंट = 0 पर शिफ्ट में कूदते हैं। ( neg cl
ZF सेट करता है, जिसे हम शिफ्ट कर सकते हैं और यदि हम शिफ्ट नहीं होते हैं तो रिटर्न वैल्यू के रूप में छोड़ सकते हैं ।) ध्यान दें कि 32-mask mod 32 = -mask
, और x86 स्केलर शिफ्टर्स अपनी गिनती को मास्क करके & 31
या & 63
।
line addr machine NASM source. (from nasm -felf64 -l/dev/stdout)
num code bytes
1 %use smartalign
2
3 ;10.4.1.33 10.4.0.0/23 true
4 ;10.4.1.33 10.4.0.0/24 false
5
6 ;; /codegolf/185005/im-in-your-subnets-golfing-your-code
7 %ifidn __OUTPUT_FORMAT__, elf64
8 in_subnet:
9
10 00000000 6A03 push 3
11 00000002 5F pop rdi ; edi = 3 dotted-quads to parse, sort of.
12 .parseloop:
13
14 ;xor ebx,ebx ; doesn't need to be zeroed first; we end up shifting out the original contents
15 ;lea ecx, [rbx+4]
16 00000003 6A04 push 4
17 00000005 59 pop rcx ; rcx = 4 integers in a dotted-quad
18 .quadloop:
19
20 00000006 31D2 xor edx,edx ; standard edx=atoi(rdi) loop terminated by a non-digit char
21 00000008 EB05 jmp .digit_entry
22 .digitloop:
23 0000000A 6BD20A imul edx, 10
24 0000000D 00C2 add dl, al
25 .digit_entry:
26 0000000F AC lodsb
27 00000010 2C30 sub al, '0'
28 00000012 3C09 cmp al, 9
29 00000014 76F4 jbe .digitloop
30 ; al=non-digit character - '0'
31 ; RDI pointing to the next character.
32 ; EDX = integer
33
34 00000016 C1E308 shl ebx, 8
35 00000019 88D3 mov bl, dl ; build a quad 1 byte at a time, ending with the lowest byte
36 0000001B E2E9 loop .quadloop
37
38 0000001D 53 push rbx ; push result to be collected after parsing 3 times
39 0000001E FFCF dec edi
40 00000020 75E1 jnz .parseloop
41
42 00000022 59 pop rcx ; /mask (at the top of a dword)
43 00000023 5A pop rdx ; subnet
44 00000024 58 pop rax ; host
45 00000025 0FC9 bswap ecx ; cl=network bits (reusing the quad parse loop left it in the high byte)
49 00000027 F6D9 neg cl
50 00000029 7404 jz .all_net ; skip the count=32 special case
51
52 0000002B 31D0 xor eax, edx ; host ^ subnet
53 0000002D D3E8 shr eax, cl ; shift out the host bits, keeping only the diff of subnet bits
54
55 .all_net:
56 ; setnz al ; return ZF=1 match, ZF=0 not in subnet
57 0000002F C3 ret
58 00000030 30 .size: db $ - in_subnet
0x30 = 48 bytes
(नवीनतम संस्करण के साथ अद्यतन नहीं)
इसे ऑनलाइन आज़माएं!
इसमें _start
वह भी शामिल है जो इसे कॉल करता है argv[1]
और एक निकास स्थिति देता है।
## on my desktop
$ ./ipv4-subnet "10.4.1.33 10.4.0.0/24" && echo "$? : in subnet" || echo "$? : not in subnet"
not in subnet
$ ./ipv4-subnet "10.4.1.33 10.4.0.0/23" && echo "$? : in subnet" || echo "$? : not in subnet"
in subnet
यह ठीक काम करता है यदि आप एक कमांड लाइन arg पास करते हैं जिसमें एक स्थान के बजाय एक नई रेखा होती है। लेकिन इसके बजाय होना है , साथ ही नहीं।
x86 32-बिट मशीन कोड फ़ंक्शन, 38 बाइट्स
9 पूर्णांक करें -> uint8_t पार्स और उन्हें स्टैक पर "पुश" करें, जहां हम उन्हें पासवर्ड के रूप में पॉप करते हैं या सीएल में अंतिम एक का उपयोग करते हैं। स्ट्रिंग के अंत में अतीत को पढ़ने से बचा जाता है।
इसके अलावा, dec
32-बिट मोड में केवल 1 बाइट है।
72 in_subnet:
73 00000000 89E7 mov edi, esp
74 00000002 51 push ecx
75 00000003 51 push ecx ; sub esp,8
76 .byteloop:
77
78 00000004 31C9 xor ecx,ecx ; standard ecx=atoi(rdi) loop terminated by a non-digit char
79 ; runs 9 times: 8 in two dotted-quads, 1 mask length
80 00000006 EB05 jmp .digit_entry
81 .digitloop:
82 00000008 6BC90A imul ecx, 10
83 0000000B 00C1 add cl, al
84 .digit_entry:
85 0000000D AC lodsb
86 0000000E 2C30 sub al, '0'
87 00000010 3C09 cmp al, 9
88 00000012 76F4 jbe .digitloop
89 ; RDI pointing to the next character.
90 ; EDX = integer
91
92 00000014 4F dec edi
93 00000015 880F mov [edi], cl ; /mask store goes below ESP but we don't reload it
94 00000017 39E7 cmp edi, esp
95 00000019 73E9 jae .byteloop
96
97 ;; CL = /mask still there from the last conversion
98 ;; ESP pointing at subnet and host on the stack, EDI = ESP-1
99
100 0000001B 5A pop edx ; subnet
101 0000001C 58 pop eax ; host
102
103 0000001D 31D0 xor eax, edx ; host ^ subnet
104 0000001F F6D9 neg cl ; -mask = (32-mask) mod 32; x86 shifts mask their count
105 00000021 7402 jz .end ; 32-n = 32 special case
106 00000023 D3E8 shr eax, cl
107 .end:
108 ; setz al ; just return in ZF
109 00000025 C3 ret
110 00000026 26 .size: db $ - in_subnet
0x26 = 38 bytes
परीक्षण करने वाला
113 global _start
114 _start:
115 00000027 8B742408 mov esi, [esp+8] ; argv[1]
116 0000002B E8D0FFFFFF call in_subnet
117 00000030 0F95C3 setnz bl
118 00000033 B801000000 mov eax, 1 ; _exit syscall
119 00000038 CD80 int 0x80