While adding a detailed answer, I noticed that GCC does not warn the following code while Visual C++ complains.
#include <cstring>
int main()
{
const char CONSTSTR[] = "foo/bar/foobar.txt";
char *nonconst = std::strrchr (CONSTSTR, '/');
// cannot convert from 'const char *' to 'char *'
*nonconst++ = 'B';
*nonconst++ = 'A';
*nonconst++ = 'D';
}
I have tested three different GCC versions:
- 4.1.2 on Red Hat (Linux)
- 4.5.3 on Cygwin (Windows)
- 4.7.2 on MinGW (Windows)
But all these three GCC versions compiled this code without any warning/error:
> g++ -Wall -Wextra -pedantic -ansi test.cpp && echo "success"
success
While Microsoft compiler v16 complains:
> cl -c test.cpp
Microsoft (R) 32-bit C/C++ Optimizing Compiler Version 16.00.30319.01 for 80x86
Copyright (C) Microsoft Corporation. All rights reserved.
test.cpp
test.cpp(5) : error C2440: 'initializing' : cannot convert from 'const char *' to 'char *'
Conversion loses qualifiers
(from my office, I do not have access to ideone/codepad/... to test it using other versions)
As this code uses std::strrchr, I do not understand why GCC does not complain.
const char* strrchr( const char* str, int ch ); //the code above uses this declaration
char* strrchr( char* str, int ch );
My question: Why does g++ successfully compile this code without any warning/error? Is it a bug? a feature? a miss-configuration on my side?
Actually your g++ does not accept the conversion from '
const char *
' to 'char *
', it's just that on your versionstd::strrchr()
returns achar*
(incorrectly, instead of aconst char*
).To verify the first part of my statement, try to compile the following on your GCC versions, I predict that all will correctly issue an error:
Now for the second part, I tried to compile the following minimal code, whose actual aim is to trigger an error in order to list the declared overloads of
std::strrchr
:Well, with gcc 4.7.2 the message shows the expected "all non-const" and "all const" overloads:
i.e. the prototypes
But with gcc 4.3.2 the message was different:
i.e. the overloads were
(the second one is the C++ non-const overload; but the first one is the old C version, and should instead be the C++ const overload).
This it seems that the headers (
<cstring>
and/or<string.h>
) were incorrect on this version, and I suspect that it's the same on yours.Edit: I found for example a discussion, a blog post and a bug report (for
strchr
notstrrchr
but it's the same story).