strange behavior of by-value function arguments in postcondition

bauss jj_1337 at live.dk
Tue Aug 31 11:33:10 UTC 2021


On Tuesday, 31 August 2021 at 11:30:16 UTC, bauss wrote:
> On Tuesday, 31 August 2021 at 10:43:41 UTC, Andrzej K. wrote:
>> Have a look at the following short example:
>>
>> https://d.godbolt.org/z/1KoofEx5Y
>>
>>
>> We have a function that takes arguments by value and has a 
>> postcondition that refers to these arguments:
>>
>> ```D
>> int select(int lo, int hi)
>>   in (lo <= hi)
>>   out (r; lo <= r && r <= hi)
>> {
>>   int ans = lo;
>>   lo = hi; // why not
>>   return ans;
>> }
>> ```
>>
>> I would intuitively assume that the contract of such a 
>> function is: "I will not modify your objects, and I will 
>> select a number from the range that your objects indicate." 
>> This expectation of mine can be reflected by the following 
>> assertions:
>>
>> ```D
>> void main()
>> {
>>     int lo = 1;
>>     int hi = 4;
>>     int i = select(lo, hi);
>>     assert(lo == 1);
>>     assert(hi == 4);
>>     assert(i >= 1 && i <= 4);
>> }
>> ```
>>
>> The fact that function `select()` modifies its copy of the 
>> argument, should not be of interest to the caller: it is an 
>> implementation detail of the function. And all the assertions 
>> in function `main()` are satisfied. However, the fact that 
>> postcondition observes the copies of user objects makes the 
>> postcondition fail.
>>
>> I realize that this effect is consistent with the mechanics of 
>> the language. But it seems that the mechanics of the language 
>> render the postconditions somewhat counterintuitive. The 
>> postcondition should reflect the contract: it should describe 
>> something that the caller can also see and understand: not the 
>> state of internal copies of user input.
>>
>> Is this not a design bug?
>
> To me it makes perfect sense as the function would probably 
> translated to something like this:
>
> int select(int lo, int hi)
> {
>   // in
>   assert(lo <= hi);
>
>   // body
>   int ans = lo;
>   lo = hi;
>
>   // out
>   int r = ans;
>   assert(lo <= r && r <= hi);
>
>   return r;
> }

Yep, I was right.

Looking at the AST this is what the compiler generates for the 
function:

```d
int select(int lo, int hi)
in (lo <= hi)
out (r; lo <= r && (r <= hi))
{
	{
		assert(lo <= hi);
	}
	int ans = lo;
	lo = hi;
	__result = ans;
	goto __returnLabel;
	__returnLabel:
	{
		const ref const(int) r = __result;
		assert(lo <= r && (r <= hi));
	}
	return __result;
}
```


More information about the Digitalmars-d mailing list