lazarus/ide/generatefppkgconfigurationdlg.pas

503 lines
16 KiB
ObjectPascal

{ Dialog to write fppkg-configuration files (fppkg.cfg and default) using the
fpcmkcfg tool that comes with fpc.
Copyright (C) 2019 Joost van der Sluis/CNOC joost@cnoc.nl
This source is free software; you can redistribute it and/or modify it under
the terms of the GNU General Public License as published by the Free
Software Foundation; either version 2 of the License, or (at your option)
any later version.
This code is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
details.
A copy of the GNU General Public License is available on the World Wide Web
at <http://www.gnu.org/copyleft/gpl.html>. You can also obtain it by writing
to the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor,
Boston, MA 02110-1335, USA.
}
unit GenerateFppkgConfigurationDlg;
{$mode objfpc}{$H+}
interface
uses
// Rtl
Classes, SysUtils,
// Fcl
fpmkunit, process,
// Fppkg
pkgglobals,
// LazUtils
FPCAdds, LazFileUtils, LazFileCache, UTF8Process,
// Lcl
Forms, Controls, Graphics, Dialogs, StdCtrls, ExtCtrls,
// Codetools
CodeToolManager, DefineTemplates,
// Ideintf
IDEDialogs,
// IdeConfig
IDEProcs, LazConf, EnvironmentOpts, FppkgHelper,
// IDE
LazarusIDEStrConsts, InitialSetupProc;
type
{ TGenerateFppkgConfigurationDialog }
TGenerateFppkgConfigurationDialog = class(TForm)
FppkgLabel: TLabel;
FpcPrefixCombobox: TComboBox;
FppkgPrefixLabel: TLabel;
InfoMemo: TMemo;
BtnPanel: TPanel;
FppkgWriteConfigButton: TButton;
WarningsLabel: TLabel;
BrowsePanel: TPanel;
BrowseButton: TButton;
procedure FormCreate(Sender: TObject);
procedure FpcPrefixComboboxChange(Sender: TObject);
procedure BrowseButtonClick(Sender: TObject);
procedure FppkgWriteConfigButtonClick(Sender: TObject);
private
FCompiler: string;
FFppkgCfgFilename: string;
fLastParsedFpcPrefix: string;
fLastParsedFpcLibPath: string;
procedure SetCompiler(AValue: string);
procedure SetFppkgCfgFilename(AValue: string);
function CheckIfWritable(Filename: string): Boolean;
function CheckFppkgQuality(APrefix: string; out LibPath, Note: string): TSDFilenameQuality;
procedure UpdateFppkgNote;
procedure SearchFppkgFpcPrefixCandidates;
function CheckFpcmkcfgQuality(out Note: string): TSDFilenameQuality;
public
// Filename of the Free Pascal compiler that has to be written to the
// configuration-files.
property Compiler: string read FCompiler write SetCompiler;
// Filename of the configuration file that has to be written.
property FppkgCfgFilename: string read FFppkgCfgFilename write SetFppkgCfgFilename;
end;
var
GenerateFppkgConfigurationDialog: TGenerateFppkgConfigurationDialog;
implementation
{$R *.lfm}
{ TGenerateFppkgConfigurationDialog }
procedure TGenerateFppkgConfigurationDialog.FormCreate(Sender: TObject);
begin
Caption := lisGenerateFppkgConfigurationCaption;
FppkgLabel.Caption := lisGenerateFppkgConfiguration;
{$IFDEF WINDOWS}
FppkgPrefixLabel.Caption:=Format(lisFppkgInstallationPath, [GetFPCVer+PathDelim+'units', GetFPCVer+PathDelim+'fpmkinst']);
{$ELSE}
FppkgPrefixLabel.Caption:=Format(lisFppkgInstallationPath, ['lib/fpc', 'lib64/fpc']);
{$ENDIF WINDOWS}
SearchFppkgFpcPrefixCandidates;
FpcPrefixCombobox.Text := '';
if FpcPrefixCombobox.Items.Count > 0 then
FpcPrefixCombobox.ItemIndex := 0;
WarningsLabel.Caption := lisFppkgConfGenProblems;
FppkgWriteConfigButton.Caption := lisFppkgWriteConfigFile;
BrowseButton.Caption:=lisPathEditBrowse;
UpdateFppkgNote;
end;
procedure TGenerateFppkgConfigurationDialog.SearchFppkgFpcPrefixCandidates;
function CheckPath(APath: string; List: TStrings): boolean;
var
LibPath, Note: String;
begin
Result:=false;
if APath='' then exit;
ForcePathDelims(APath);
// check if already checked
if Assigned(List) and (List.IndexOf(APath)>-1) then exit;
if CheckFppkgQuality(APath, LibPath, Note) = sddqCompatible then
begin
List.Add(APath);
Result := True;
end;
end;
var
ChkPath: string;
begin
FpcPrefixCombobox.Clear;
ChkPath := ExtractFileDir(ExtractFileDir(EnvironmentOptions.GetParsedCompilerFilename));
{$IFDEF WINDOWS}
ChkPath := ExtractFileDir(ChkPath);
{$ENDIF WINDOWS}
CheckPath(ChkPath, FpcPrefixCombobox.Items);
{$IFNDEF WINDOWS}
// Check if the user provided the compiler-executable inside the lib-directory
// itself. (prefix/lib/3.3.1/ppcarm or something)
ChkPath := ExtractFileDir(ExtractFileDir(ChkPath));
CheckPath(ChkPath, FpcPrefixCombobox.Items);
{$ENDIF}
{$IFDEF WINDOWS}
CheckPath('C:\PP', FpcPrefixCombobox.Items);
CheckPath('D:\PP', FpcPrefixCombobox.Items);
CheckPath('C:\FPC', FpcPrefixCombobox.Items);
CheckPath('D:\FPC', FpcPrefixCombobox.Items);
{$ELSE}
CheckPath('/usr/lib/fpc', FpcPrefixCombobox.Items);
CheckPath('/usr/lib64/fpc', FpcPrefixCombobox.Items);
{$IFDEF Linux}
CheckPath('/usr/lib/'+FPCAdds.GetCompiledTargetCPU+'-linux-gnu/fpc/default', FpcPrefixCombobox.Items);
{$IFDEF CPUARM}
CheckPath('/usr/lib/'+FPCAdds.GetCompiledTargetCPU+'-linux-gnueabi/fpc/default', FpcPrefixCombobox.Items);
CheckPath('/usr/lib/'+FPCAdds.GetCompiledTargetCPU+'-linux-gnueabihf/fpc/default', FpcPrefixCombobox.Items);
{$ENDIF}
{$ENDIF}
CheckPath('/usr/local/lib/fpc', FpcPrefixCombobox.Items);
CheckPath('/usr/local/lib64/fpc', FpcPrefixCombobox.Items);
{$ENDIF WINDOWS}
end;
function TGenerateFppkgConfigurationDialog.CheckFppkgQuality(APrefix: string; out LibPath,
Note: string): TSDFilenameQuality;
var
SR: TRawByteSearchRec;
LibPathValid: Boolean;
Ver: TFPVersion;
begin
Result := sddqInvalid;
LibPath := '';
if APrefix='' then
begin
Note := lisWarning + lisNoFppkgPrefix + LineEnding;
Exit;
end;
APrefix:=TrimFilename(APrefix);
if not FileExistsCached(APrefix) then
begin
Note:= lisWarning + lisFreePascalPrefix + ' ' + lisISDDirectoryNotFound + '.' + LineEnding;
end
else if not DirPathExistsCached(APrefix) then
begin
Note:= lisWarning + lisFreePascalPrefix + ' ' + lisPathIsNoDirectory + LineEnding;
end
else
begin
LibPath := AppendPathDelim(APrefix);
LibPathValid := True;
if DirPathExistsCached(LibPath+PathDelim+'fpmkinst') and
DirPathExistsCached(LibPath+PathDelim+'units') then
begin
LibPathValid := True;
Result := sddqCompatible;
end
else if LibPathValid and (FindFirstUTF8(LibPath+AllFilesMask, faDirectory, SR) = 0) then
begin
LibPathValid := False;
repeat
if (SR.Name<>'.') and (SR.Name<>'..') then
begin
if DirPathExistsCached(LibPath+SR.Name+PathDelim+'fpmkinst') and
DirPathExistsCached(LibPath+SR.Name+PathDelim+'units') then
begin
Ver := TFPVersion.Create;
try
Ver.AsString:=SR.Name;
if (Ver.Major > -1) and (Ver.Minor > -1) and (Ver.Micro > -1) then
LibPath:=LibPath + '{CompilerVersion}' + PathDelim
else
LibPath:=LibPath + SR.Name + PathDelim
finally
Ver.Free;
end;
LibPathValid := True;
Result := sddqCompatible;
Break;
end;
end;
until FindNext(SR) <> 0;
FindCloseUTF8(SR);
end;
if not LibPathValid then
Note:= Note + lisWarning + lisNotAValidFppkgPrefix + LineEnding
else
Note:='';
end;
end;
procedure TGenerateFppkgConfigurationDialog.UpdateFppkgNote;
var
CurCaption: String;
Msg, Note: string;
FileName: string;
begin
if csDestroying in ComponentState then exit;
CurCaption:=FpcPrefixCombobox.Text;
if (fLastParsedFpcPrefix=CurCaption) and (CurCaption<>'') then exit;
fLastParsedFpcPrefix:=CurCaption;
Msg := '';
if CheckFppkgQuality(CurCaption,fLastParsedFpcLibPath,Note)<>sddqCompatible then
Msg := Note;
if (CheckFPCExeQuality(FCompiler, Note, CodeToolBoss.CompilerDefinesCache.TestFilename)<>sddqCompatible) then
Msg := Msg + lisWarning + lisFppkgCompilerProblem +Note + LineEnding;
if CheckFpcmkcfgQuality(Note) <> sddqCompatible then
Msg := Msg + lisWarning + Note + LineEnding;
Note := lisFppkgFilesToBeWritten + LineEnding;
Note := Note + Format(lisGenerateFppkgCfg, [FppkgCfgFilename]) + LineEnding;
// These are the default config-locations used by fpcmkcfg
{$IFDEF WINDOWS}
FileName := '%LocalAppData%\FreePascal\Fppkg\config\default';
{$ELSE}
FileName := '~/.fppkg/config/default';
{$ENDIF}
Note := Note + Format(lisGenerateFppkgCompCfg, [FileName]) + LineEnding;
if not CheckIfWritable(FppkgCfgFilename) then
Msg := Msg + lisWarning + ueFileROText1 + FppkgCfgFilename + ueFileROText2 + LineEnding;
if not CheckIfWritable(FileName) then
Msg := Msg + lisWarning + ueFileROText1 + FileName + ueFileROText2 + LineEnding;
if Msg<>'' then
begin
WarningsLabel.Visible := True;
Note := Msg + LineEnding + Note;
FppkgWriteConfigButton.Enabled := False;
end
else
begin
WarningsLabel.Visible := False;
FppkgWriteConfigButton.Enabled := True;
end;
if fLastParsedFpcLibPath<>'' then
begin
// If the fLastParsedFpcLibPath is empty, these two lines contain garbage
Note := Note + LineEnding + Format(lisFppkgPrefix, [fLastParsedFpcPrefix]) + LineEnding;
Note := Note + Format(lisFppkgLibPrefix, [fLastParsedFpcLibPath]) + LineEnding;
end;
InfoMemo.Text := Note;
end;
procedure TGenerateFppkgConfigurationDialog.SetCompiler(AValue: string);
begin
if FCompiler = AValue then Exit;
FCompiler := AValue;
fLastParsedFpcPrefix := ' ';
UpdateFppkgNote;
end;
function TGenerateFppkgConfigurationDialog.CheckFpcmkcfgQuality(out Note: string): TSDFilenameQuality;
var
FpcmkcfgExecutable: string;
Proc: TProcessUTF8;
S: string;
Ver: TFPVersion;
begin
Result := sddqCompatible;
Note:='';
FpcmkcfgExecutable := FindFPCTool('fpcmkcfg'+GetExecutableExt, EnvironmentOptions.GetParsedCompilerFilename);
if FpcmkcfgExecutable = '' then
begin
Note := lisFppkgFpcmkcfgMissing + ' ' + lisFppkgRecentFpcmkcfgNeeded;
Result := sddqInvalid;
end
else
begin
Proc := TProcessUTF8.Create(nil);
try
Proc.Options := proc.Options + [poNoConsole, poWaitOnExit,poUsePipes];
// Write fppkg.cfg
Proc.Executable := FpcmkcfgExecutable;
proc.Parameters.Add('-V');
proc.Execute;
if proc.ExitStatus <> 0 then
begin
Note := lisFppkgFpcmkcfgCheckFailed + ' ' + lisFppkgFpcmkcfgProbTooOld + ' ' + lisFppkgRecentFpcmkcfgNeeded;
Result := sddqInvalid;
end
else
begin
S := '';
SetLength(S, Proc.Output.NumBytesAvailable);
Proc.Output.Read(S[1], Proc.Output.NumBytesAvailable);
Ver := TFPVersion.Create;
try
S := Copy(S, pos(':', S)+2);
Ver.AsString := Trim(S);
if Ver.Major = -1 then
begin
Note := lisFppkgFpcmkcfgCheckFailed + ' ' + lisFppkgFpcmkcfgNeeded + lisFppkgRecentFpcmkcfgNeeded;
Result := sddqInvalid;
end
else if not ((Ver.Major = 0) or (Ver.Major > 3) or (((Ver.Major = 3)) and (Ver.Minor>1))) then
begin
// fpcmkcfg's version must be > 3.1. Older versions need other
// parameters. Version 0 is also allowed, because it is probably
// self-built.
Note := Format( lisFppkgFpcmkcfgTooOld, [Ver.AsString]) + ' ' + lisFppkgFpcmkcfgNeeded + ' ' + lisFppkgRecentFpcmkcfgNeeded;
Result := sddqInvalid;
end;
finally
Ver.Free;
end;
end;
finally
Proc.Free;
end;
end;
end;
procedure TGenerateFppkgConfigurationDialog.FpcPrefixComboboxChange(Sender: TObject);
begin
UpdateFppkgNote;
end;
procedure TGenerateFppkgConfigurationDialog.SetFppkgCfgFilename(AValue: string);
begin
if FFppkgCfgFilename = AValue then Exit;
FFppkgCfgFilename := AValue;
fLastParsedFpcPrefix := ' ';
UpdateFppkgNote;
end;
procedure TGenerateFppkgConfigurationDialog.BrowseButtonClick(Sender: TObject);
var
Dlg: TSelectDirectoryDialog;
begin
Dlg:=TSelectDirectoryDialog.Create(nil);
try
Dlg.Title:=lisSelectFPCPath;
Dlg.Options:=Dlg.Options+[ofPathMustExist];
if not Dlg.Execute then exit;
FpcPrefixCombobox.Text:=Dlg.FileName;
finally
Dlg.Free;
end;
UpdateFppkgNote;
end;
procedure TGenerateFppkgConfigurationDialog.FppkgWriteConfigButtonClick(Sender: TObject);
var
Msg: string;
FpcmkcfgExecutable, CompConfigFilename: string;
Proc: TProcessUTF8;
Fppkg: TFppkgHelper;
procedure ShowFpcmkcfgError;
begin
SetLength(Msg, Proc.Output.NumBytesAvailable);
if Msg <> '' then
begin
Proc.Output.Read(Msg[1], Proc.Output.NumBytesAvailable);
IDEMessageDialog(lisFppkgProblem, Format(lisFppkgCreateFileFailed, [GetFppkgConfigFile(False, False), Msg]), mtWarning, [mbOK])
end;
end;
begin
try
FpcmkcfgExecutable := FindFPCTool('fpcmkcfg'+GetExecutableExt, EnvironmentOptions.GetParsedCompilerFilename);
if FpcmkcfgExecutable<>'' then
begin
Proc := TProcessUTF8.Create(nil);
try
Proc.Options := proc.Options + [poWaitOnExit, poNoConsole, poUsePipes, poStderrToOutPut];
// Write fppkg.cfg
Proc.Executable := FpcmkcfgExecutable;
proc.Parameters.Add('-p');
proc.Parameters.Add('-3');
proc.Parameters.Add('-o');
proc.Parameters.Add(FppkgCfgFilename);
proc.Parameters.Add('-d');
proc.Parameters.Add('globalpath='+fLastParsedFpcLibPath);
proc.Parameters.Add('-d');
{$IFDEF WINDOWS}
proc.Parameters.Add('globalprefix='+fLastParsedFpcLibPath);
{$ELSE}
proc.Parameters.Add('globalprefix='+fLastParsedFpcPrefix);
{$ENDIF}
proc.Execute;
Fppkg:=TFppkgHelper.Instance;
if proc.ExitStatus <> 0 then
ShowFpcmkcfgError
else
begin
Fppkg:=TFppkgHelper.Instance;
Fppkg.ReInitialize;
// Write default compiler configuration file
CompConfigFilename := Fppkg.GetCompilerConfigurationFileName;
if CompConfigFilename <> '' then
begin
proc.Parameters.Clear;
proc.Parameters.Add('-p');
proc.Parameters.Add('-4');
proc.Parameters.Add('-o');
proc.Parameters.Add(CompConfigFilename);
proc.Parameters.Add('-d');
proc.Parameters.Add('fpcbin='+EnvironmentOptions.GetParsedCompilerFilename);
proc.Execute;
if proc.ExitStatus <> 0 then
ShowFpcmkcfgError
end;
end;
Fppkg.ReInitialize;
finally
Proc.Free;
end;
end;
except
on E: Exception do
IDEMessageDialog(lisFppkgProblem, Format(lisFppkgWriteConfException, [E.Message]), mtWarning, [mbOK]);
end;
fLastParsedFpcPrefix := '';
UpdateFppkgNote;
if CheckFppkgConfiguration(FFppkgCfgFilename, Msg)<>sddqCompatible then
begin
IDEMessageDialog(lisFppkgProblem, Format(lisFppkgWriteConfFailed, [Msg]),
mtWarning, [mbOK]);
ModalResult := mrCancel;
end
else
ModalResult := mrOK;
end;
function TGenerateFppkgConfigurationDialog.CheckIfWritable(Filename: string): Boolean;
begin
Result := True;
if (FileName<>'') then
begin
Filename := ExpandFileNameUTF8(Filename);
if FileExistsUTF8(Filename) then
Result := FileIsWritable(FileName)
end;
end;
end.