आपको .NET 4 x86 के घबराने में एक कोड जेनरेशन बग मिला। यह एक बहुत ही असामान्य है, यह केवल तब विफल होता है जब कोड अनुकूलित नहीं होता है। मशीन कोड इस तरह दिखता है:
State a = s[0, 0];
013F04A9 push 0 ; index 2 = 0
013F04AB mov ecx,dword ptr [ebp-40h] ; s[] reference
013F04AE xor edx,edx ; index 1 = 0
013F04B0 call 013F0058 ; eax = s[0, 0]
013F04B5 mov dword ptr [ebp-4Ch],eax ; $temp1 = eax
013F04B8 movsx eax,byte ptr [ebp-4Ch] ; convert sbyte to int
013F04BC mov dword ptr [ebp-44h],eax ; a = s[0, 0]
Console.WriteLine(a == s[0, 0]); // False
013F04BF mov eax,dword ptr [ebp-44h] ; a
013F04C2 mov dword ptr [ebp-50h],eax ; $temp2 = a
013F04C5 push 0 ; index 2 = 0
013F04C7 mov ecx,dword ptr [ebp-40h] ; s[] reference
013F04CA xor edx,edx ; index 1 = 0
013F04CC call 013F0058 ; eax = s[0, 0]
013F04D1 mov dword ptr [ebp-54h],eax ; $temp3 = eax
; <=== Bug here!
013F04D4 mov eax,dword ptr [ebp-50h] ; a == s[0, 0]
013F04D7 cmp eax,dword ptr [ebp-54h]
013F04DA sete cl
013F04DD movzx ecx,cl
013F04E0 call 731C28F4
बहुत सारे टेम्पोररी और कोड डुप्लीकेशन के साथ एक प्लोडिंग अफेयर, यह अडॉप्ट किए गए कोड के लिए सामान्य है। 013F04B8 पर निर्देश उल्लेखनीय है, यही वह जगह है जहां 32-बिट पूर्णांक से sbyte से आवश्यक रूपांतरण होता है। सरणी गेट्टर हेल्पर फ़ंक्शन स्टेट के बराबर 0x0000000FF लौटा दिया। बग, और मान को तुलना करने से पहले इसे -1 (0xFFFFFFFFFF) में बदलने की आवश्यकता है। MOVSX इंस्ट्रक्शन एक साइन एक्सटेन्शन इंस्ट्रक्शन है।
समान बात 013F04CC पर फिर से होती है, लेकिन इस बार एक ही रूपांतरण करने के लिए कोई MOVSX निर्देश नहीं है । जहां चिप्स नीचे गिरता है, सीएमपी निर्देश 0xFFFFFFFF की तुलना 0x000000FF से करता है और यह गलत है। तो यह चूक की एक त्रुटि है, कोड जेनरेटर MOVSX को फिर से वही रूपांतरण करने के लिए एमवायएसएक्स से बाहर निकलने में विफल रहा।
इस बग के बारे में विशेष रूप से असामान्य है कि यह सही ढंग से काम करता है जब आप अनुकूलक को सक्षम करते हैं, तो अब यह दोनों मामलों में MOVSX का उपयोग करना जानता है।
संभावित कारण यह है कि यह बग इतने लंबे समय के लिए अनिच्छुक हो गया था क्योंकि यह ईनम का आधार प्रकार है। काफी दुर्लभ है। बहु-आयामी सरणी का उपयोग करना महत्वपूर्ण है और साथ ही, संयोजन घातक है।
नहीं तो एक बहुत महत्वपूर्ण बग मैं कहूँगा। कितना व्यापक हो सकता है यह अनुमान लगाना कठिन है, मेरे पास परीक्षण करने के लिए केवल 4.6.1 x86 घबराना है। X64 और 3.5 x86 घबराना बहुत अलग कोड उत्पन्न करते हैं और इस बग से बचते हैं। अस्थायी रूप से चलते रहने का तरीका यह है कि Enum बेस टाइप के रूप में sbyte को हटा दिया जाए और इसे डिफ़ॉल्ट, int के रूप में होने दिया जाए , इसलिए कोई साइन एक्सटेंशन आवश्यक नहीं है।
आप बग को connect.microsoft.com पर दर्ज कर सकते हैं, इस Q + A से लिंक करना उन्हें सब कुछ बताने के लिए पर्याप्त होना चाहिए जो उन्हें जानना आवश्यक है। मुझे बताएं कि क्या आप समय नहीं लेना चाहते हैं और मैं इसका ध्यान रखूंगा।