User Changes Trunk
- 1 About this page
- 2 All systems
- 2.1 Unit changes
- 2.1.1 FileNameCaseSensitive and FileNameCasePreserving in unit System
- 2.1.2 Several methods of TDataset changes signature (TRecordBuffer)
- 2.1.3 DLLParam changed from Longint into PtrInt
- 2.1.4 TBookmark TBookmarkstr and TDataset.Bookmark change to Delphi2009+ compatible definitions
- 2.1.5 Some longtime deprecated functions in unit Unix have been removed.
- 2.1.6 Some symbols in unit Unix and Unixutils have been deprecated
- 2.2 Language Changes
- 2.1 Unit changes
- 3 See also
About this page
Below you can find a list of intentional changes since the previous release that can change the behaviour of previously working code, along with why these changes were performed and how you can adapt your code if you are affected by them. The list of new features can be found here.
FileNameCaseSensitive and FileNameCasePreserving in unit System
- Old behaviour: There was only one constant (FileNameCaseSensitive) which signalized behaviour of the respective target platform with regard to the case in filenames. This was often set to true also on platforms not really treating filenames in case sensitive manner (e.g. Win32 and Win64).
- New behaviour: FileNameCaseSensitive is set to true only on platforms which really treat filenames in case sensitive manner and FileNameCasePreserving constant is added to unit system of all platforms to signalize whether the case in filenames supplied when creating or renaming them is preserved or not. This might possibly break existing user code relying on the previous (sometimes incorrect) value of FileNameCaseSensitive.
- Reason: In reality, there are two distinct types of behaviour. If a platform is case sensitive, searching for file "a" will never result in finding file "A" whereas case insensitive platform will happily return "A" if such a file is found on the disk. If a platform is case preserving, it will store file "a" as "a" on disk whereas case non-preserving platform may convert the filenames to uppercase before storing them on disk (i.e. file "a" is in reality stored as "A"). Case non-preserving platforms are never case sensitive, but case preserving platforms may or may not be case sensitive at the same time.
- Remedy: Review your existing code using FileNameCaseSensitive, check which of the two properties described above fit the particular purpose in your case and change to FileNameCasePreserving where appropriate.
Several methods of TDataset changes signature (TRecordBuffer)
- Old behaviour: Several (virtual) methods of TDataset have parameters of type "pchar", which are often called "buffer".
- New behaviour: The pchar type has been changed to TRecordBuffer. Currently this type is still an alias for p(ansi)char, but in time it will be changed to pbyte for the 2.7.1/2.8.0 branch, which is D2009+ compatible.
- Reason: Preparation for Delphi 2009+ compatibility and improving of general typing. In Delphi 2009+ (and fully compatible FPC modes in the future) pchar is not pointer to byte anymore. This change will be merged back to 2.6(.2), but with TRecordBuffer=pchar.
- Remedy: Change the relevant virtual methods to use TRecordBuffer for buffer parameters. Define TRecordBuffer=pansichar to keep older Delphis and FPCs working. In places where a buffer is typecasted, don't use pchar but the symbol TRecordbuffer.
DLLParam changed from Longint into PtrInt
- Old behaviour: DLLParam was of type Longint even on Win64.
- New behaviour: DLLParam is now of type PtrInt so also on 64 Bit systems.
- Reason: Prevent data loss, match the declaration in the Windows headers.
- Remedy: Change the declaration of the procedures used as dll hook to take a PtrInt parameter instead of Longint.
TBookmark TBookmarkstr and TDataset.Bookmark change to Delphi2009+ compatible definitions
- Old behaviour: TDataset.Bookmark was of type TBookmarkstr, TBookmarkstr=string, Tbookmark=pointer
- New behaviour: TDataset.Bookmark is of type TBookmark, TBookmarkstr=ansistring, TBookmark=TBytes
- Reason: D2009+ compatibility, where probably the fact that string became a 2-byte encoding was the trigger.
- Remedy: Adjust typing in descendents and using code accordingly.
Some longtime deprecated functions in unit Unix have been removed.
- Old behaviour: Unix.(f)statfs, Unix.shell and Unix.fsync symbols existed in unit Unix
- New behaviour: Unix.(f)statfs, Unix.shell and Unix.fsync were removed
- Reason: These were helper functions and leftovers from the 1.0.x->2.0.0 conversion, deprecated since 2007. Shell had non standarized conventions
- Remedy: All but shell: use the same functions with a "fp" prefix. Shell: use fpsystem. If you checked the return value for other values than -1, a modification of the errorchecking might be necessary. Refer to your *nix documention, or have a look at the transformfpsystemtoshell function in tests/util/redir.pp in the FPC sources to see what a quick fix looks like.
Some symbols in unit Unix and Unixutils have been deprecated
- Old behaviour: No deprecated warning for unixutils.getfs (several variants), unix.fpsystem(shortstring version only), Unix.MS_ constants and unix.tpipe
- New behaviour: The compiler will emit a deprecated warning for these symbols. In future versions these may be removed.
- Reason: getfs has been replaced by a wholly cross-platform function sysutils.getfilehandle long ago. fpsystem(shortstring) was a leftover of the 1.0.x->2.0.x migration (the ansistring version remains supported), the MS_ constants are for an msync call that is not supported by FPC, and thus have been unused and unchecked (for e.g. portability aspects), tpipe was the 1.0.x alias of baseunix.TFildes, the unit where the (fp)pipe was moved to in during 2.0 series.
- Remedy: Use the new variants(sysutils.getfilehandle,fpsystem(ansistring),baseunix.tfildes). In the case of the MS_ constants, obtain current values for the constants from the same place where you got the code that uses them.
Anonymous inherited calls
- Old behaviour: An anonymous inherited call could call through to any method in a parent class that accepted arguments compatible to the parameters of the current method.
- New behaviour: An anonymous inherited call is guaranteed to always call through to the method in a parent class that was overridden by the current one.
- Example: See http://svn.freepascal.org/svn/fpc/trunk/tests/tbs/tb0577.pp. In previous FPC versions, the inherited call in tc3.test would call through to tc2.test(b: byte; l: longint = 1234);. Now it calls through to tc.test.
- Reason: Conform to the FPC documentation, Delphi-compatibility.
- Remedy: If you wish the compiler to decide which method to call based on the specified parameters, use a fully specified inherited call expression such as inherited test(b).
Overload modifier must be present in the interface
- Old behaviour: It was possible to declare a function/procedure/method as overload only in the implementation.
- New behaviour: If an overload directive is used, it must also appear in the interface.
- Reason: The old mechanism could cause hard to find problems (depending on whether or not the implementation was already parsed, the compiler would treat the routine as if it were declared with/without overload), it could cause unwanted unit recompilations due to interface crc changes, and Delphi compatibility.
- Remedy: Make sure that the overload modifier is present both in the interface and in the implementation if you use it.
Variant overload preference for string types
- Old behaviour: Preference for string types was (from better to worse): ShortString, AnsiString, WideString, UnicodeString
- New behaviour: Preference for string types now (from better to worse): WideString, UnicodeString, AnsiString, ShortString
- Reason: Unicode characters and codepage information will not lose during the conversion and unicode Delphi compatibility.
- Remedy: Use explicit conversion (e.g. ShortString(V)) if it is needed to pass variant to routine with paticular string type argument.
Default values in implementation but not in interface/forward declaration
- Old behaviour: The implementation of a routine could declare default values even if the interface declaration did not have any.
- New behaviour: Default values, if any, must be specified in the interface. They can be repeated in the implementation, but that is not required (just like it was not in the past).
- Reason: Default parameters specified only in the implementation are not, and cannot, be propagated to the interface since a unit's interface can be used before the implementation has been parsed (see http://bugs.freepascal.org/view.php?id=19434), Delphi compatibility
- Remedy: Always specify default parameter values (also) in the interface/forward declaration of a routine.