Oxygen Basic
Programming => Problems & Solutions => Topic started by: jcfuller on July 17, 2019, 06:44:31 AM
-
Charles,
My big test app appears fine now but still issues.
Did you mistakenly not change the version?
Now I have two 0.2.3 versions both with different issues which makes it hard to test and report.
I always change my version numbers for even the most minor fixes.
Attached works fine with 0.2.2
James
-
James,
Thanks for your code. I've fixed the problem now. Earlier 0.2.3 are toast, and 0.2.4 is soon to arrive.
-
In my tests, this compiles and runs fine.
-
I need help with this one though....
Code:
% SND_SYNC = 0
% SND_RESOURCE = 262148
DECLARE FUNCTION SNDPLAYSOUND LIB "WINMM.DLL" ALIAS "sndPlaySoundA" (azsSound AS ASCIIZ, BYVAL Opts AS DWORD) AS INT
DECLARE FUNCTION PBMAIN() AS INT
FUNCTION PBMAIN() AS INT
IF SNDPLAYSOUND(bycopy "WAV0001", SND_RESOURCE OR SND_SYNC) THEN
print "Sound played"
ELSE
print "Could not play sound"
END IF
END FUNCTION
PBMAIN() ' invoke entry point
Resource:
WAV0001 WAVE "C:\\executing.wav"
Also tried SOUND instead of WAVE.
Compiles fine, also resources. Resource link OK, Runs OK... but takes a long time, no sound and prints: Could not play sound
WAV file attached.
-
Hi Brian,
It plays directly. The resource thing may be obsolete:
https://docs.microsoft.com/en-us/previous-versions/dd798676(v%3Dvs.85)
% SND_SYNC = 0
% SND_RESOURCE = 262148
DECLARE FUNCTION SNDPLAYSOUND LIB "WINMM.DLL" ALIAS "sndPlaySoundA" (azsSound AS ASCIIZ, BYVAL Opts AS DWORD) AS INT
DECLARE FUNCTION PBMAIN() AS INT
FUNCTION PBMAIN() AS INT
'IF SNDPLAYSOUND(bycopy "WAV0001", SND_RESOURCE OR SND_SYNC) THEN
IF SNDPLAYSOUND(bycopy "executing.wav", SND_SYNC) THEN
print "Sound played"
ELSE
print "Could not play sound"
END IF
END FUNCTION
PBMAIN() ' invoke entry point
-
IF SNDPLAYSOUND(bycopy "executing.wav", SND_SYNC) THEN
Nope... doesnt play. :(
Does it play on your side?
-
With version 0.2.4 the app plays "Executing script" and there is a message: "Sound played". I was surprised that there is also the message "Sound played" if I rename executing.wav or if the file does not exist at all. But the docs indicate that the function plays the default sound then if possible. This was the case with my system.
-
I've done some deep digging:
Using ResEdit, I found the first letter of the resource name was omitted by GoRc.
This may be a bug caused by user-defined types (WAVE).
So I used this resource script (with "executing.wav" in the cureent directory):
// TEST
uWAV0001 WAVE c:executing.wav
resource linkage with tlink.bat:
..\..\tools\gorc /r t.rc
..\..\tools\linkres2exe t.res t.exe
pause
This now works:
ps: also works with rtl64
$filename "t.exe"
uses rtl32
% SND_SYNC = 0
% SND_RESOURCE = 262148 '0x40004
DECLARE FUNCTION PLAYSOUND LIB "WINMM.DLL" ALIAS "PlaySoundA" (azsSound AS ASCIIZ, BYVAL hModule as SYS, BYVAL Opts AS DWORD) AS INT
DECLARE FUNCTION PBMAIN() AS INT
FUNCTION PBMAIN() AS INT
sys h=GetModuleHandle(0)
IF PLAYSOUND("WAV0001", h, SND_RESOURCE OR SND_SYNC) THEN
print "Sound played"
ELSE
print "Could not play sound"
END IF
END FUNCTION
-
I also tested with our earlier version of GORC and saw the same problem. It only affects user-defined types, and appears consistent in removing the first character of the resource name.
-
Charles,
I've been playing with this and cannot get it to work using an ID instead of a string name and RCDATA instead of WAVE.
what is O2 translation of this :
#define MAKEINTRESOURCEA(i) (LPSTR)((ULONG_PTR)((WORD)(i)))
James
-
Hi James,
It casts an integer as a pseudo string-pointer.
The function being called with such a parameter detects that it is a fake pointer (having a value of less than 64k), and treats it as an equate.
Terrible hack to save on system calls!
in o2, it is sufficient to cast parameter i as a char*.
Ensure that i is an int or dword so that it has its upper 16 bits null. It may need to be a sys type for 64bit use.
(char*) i
function f(char*s){}
sys i
#show f( (char*) i )
-
Charles,
The test code should really not work. You need both the resource type and the resource ID (or name).
This is the translated code for a bc9 example.
James
void PlayRcSound (HINSTANCE hInst, char* SndName)
{
HRSRC hr = {0};
HGLOBAL hg = {0};
LPVOID lpSndData = {0};
hr = FindResource( hInst, SndName, RT_RCDATA);
if(hr != 0 )
{
hg = LoadResource( hInst, hr);
if(hg != 0 )
{
lpSndData = LockResource( hg);
if(lpSndData != 0 )
{
PlaySound((LPCTSTR)lpSndData, 0, SND_MEMORY);
return;
}
}
}
}
LRESULT CALLBACK WndProc (HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
{
if(Msg == WM_COMMAND )
{
if(LOWORD(wParam) == ID_Button1 )
{
PlayRcSound(hInstance, MAKEINTRESOURCE(12345));
}
goto L1001;
}
if(Msg == WM_DESTROY )
{
PostQuitMessage(0);
return 0;
}
L1001:
;
return DefWindowProc(hWnd, Msg, wParam, lParam);
}
-
Apparently PlaySound is smart enough to do it all in one line:
https://docs.microsoft.com/en-gb/windows/win32/multimedia/playing-wave-resources
C++
PlaySound("SoundName", hInst, SND_RESOURCE | SND_ASYNC);
-
Hey guys, I fell asleep last night, thanks for the responses.
No matter what i try i cannot get it to play a sound. I copy paste the code and it doesnt make a sound.
So it must be some of this reasons:
- how i compile the resources
- how i link the resources
- my oxygen's version having an issue (yesterday's version)
- my stupid antivirus (again)
This is my compilation log (with some notes added after generation):
PluriBASIC® 6.0.237326.0 for Oxygen BASIC for Windows
=================STEP 1 / 3 =====================
COMMANDS:
EXECUTABLE: C:\Users\Diamante\Documents\PluriBASIC\Oxygen\Exodus.exe (NOTE: Exodus is a custom DLL invoker, works fine)
WORKING DIR: C:\Users\Diamante\Documents\PluriBASIC\Clean
COMMANDLINE: c:\users\diamante\documents\pluribasic\Temp\Sound.o2bas
TIMEOUT: 300
TAKEN: 0.61
OUTPUT:
SUCCESS COMPILING
=================STEP 2 / 3 =====================
EXECUTABLE: C:\Users\Diamante\Documents\PluriBASIC\Oxygen\GoRC.exe
WORKING DIR: C:\Users\Diamante\Documents\PluriBASIC\Clean
COMMANDLINE: /r "C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.RC"
TIMEOUT: 300
TAKEN: 0.02
OUTPUT:
GoRC.Exe Version 1.0.2.1 Copyright Jeremy Gordon 1998-2019 info@goprog.com
Warnings ......................
Line 2 of Resource Script (C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.RC):-
An ID was not evaluated and was assumed to be a name:-
AAAAAAAAAAAAA_APPICON (NOTE: This works, the executable displays the icon correctly in the windows explorer)
Line 4 of Resource Script (C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.RC):-
An ID was not evaluated and was assumed to be a name:-
xWAV0001 (NOTE: I added an x before the name as Charles suggested)
Line 4 of Resource Script (C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.RC):-
A resource type was not recognised and assumed to be user-defined:-
WAVE
Output file: C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.res
=================STEP 3 / 3 =====================
EXECUTABLE: C:\Users\Diamante\Documents\PluriBASIC\Oxygen\GoRL.exe (NOTE: I changed the name of the linker to GoRL.exe, i dont remember why, but it works)
WORKING DIR: C:\Users\Diamante\Documents\PluriBASIC\Clean
COMMANDLINE: "C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.RES" "C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.exe"
TIMEOUT: 300
TAKEN: 0.02
OUTPUT: ** no console output **
This is my resource file:
AAAAAAAAAAAAA_APPICON ICON DISCARDABLE "C:\\Users\\Diamante\\Documents\\PluriBASIC\\Clean\\appicon.ico"
xWAV0001 WAVE "C:\\Users\\Diamante\\Documents\\PluriBASIC\\Clean\\executing.wav"
This is my code (With some bells and whistles added by pluribasic):
'Generated with PluriBASIC 6.0.237326.0
$ filename "C:\Users\Diamante\Documents\PluriBASIC\Clean\Sound.exe"
uses rtl32
uses console
DIM STRING ¤SYSTEM_UDT_OFFSETS(0)
STRING ¤TMPS = "" ' a temporary string.
DECLARE FUNCTION ¤GetLastError Lib "Kernel32.dll" Alias "GetLastError" () AS LONG
DECLARE FUNCTION ¤GetAsyncKeyState Lib "User32.dll" Alias "GetAsyncKeyState" (ByVal vKey AS LONG) AS short
DECLARE SUB ¤Sleep lib "Kernel32.dll" alias "Sleep" (dword mSec)
function ¤INI_QUAD(dword v1, v2) as quad
quad v = 0
copy @v+0, @v2, 4
copy @v+4, @v1, 4
return v
end function
DECLARE FUNCTION ¤OpenProcess Lib "KERNEL32.DLL" Alias "OpenProcess" (ByVal dwDesiredAccess AS DWORD, ByVal bInheritHandle AS LONG, ByVal dwProcessId AS SYS) AS SYS
DECLARE FUNCTION ¤TerminateProcess Lib "KERNEL32.DLL" Alias "TerminateProcess" ( ByVal hProcess AS SYS, ByVal uExitCode AS DWORD) AS LONG
DECLARE FUNCTION ¤CloseHandle Lib "KERNEL32.DLL" Alias "CloseHandle" (ByVal hObject AS SYS) AS LONG
DECLARE FUNCTION ¤GetCurrentProcessId Lib "KERNEL32.DLL" Alias "GetCurrentProcessId" () AS SYS
MACRO ¤SET_ERR(n)
Err.err = n
Err.erl = Err.erp
END MACRO
MACRO ¤ONERR(l, e)
Err.err = e
IF (Err.err>0) THEN
Err.ers = Err.erp
Err.erl = l
IF Err.Oe1 THEN
JMP Err.Oe1
ELSEIF Err.Oe2 THEN
CALL Err.Oe2
END IF
else
Err.ers = ""
Err.erl = 0
END IF
END MACRO
MACRO ERRCLEAR
Err.err = 0
Err.erl = 0
Err.ers = ""
END MACRO
CLASS ¤SYSERR
public sys Oe1 = 0
public sys Oe2 = 0
public int err = 0
public int erl = 0
public string erp = ""
public string ers = ""
END CLASS
' STARTS PLURIBASIC_PREPARE.BIN
' This code is executed before anything else, if you want to do something after defining other things, see PLURIBASIC_INIT
int ¤i = 0
' STARTS SYSTEM_OPERATORS.BIN
FUNCTION ¤BytOvf(byte b) AS byte
return b
END FUNCTION
FUNCTION ¤ISTRUE(byval quad v1) AS QUAD
IF v1 = 0 then
return 0
ELSE
return -1
END IF
END FUNCTION
FUNCTION ¤ISFALSE(byval quad v1) AS QUAD
IF v1 = 0 then
return -1
ELSE
return 0
END IF
END FUNCTION
FUNCTION ¤NOT(byval quad v1) AS QUAD
dword w1
dword w2
quad r
copy @w1, @v1, 4
copy @w2, @v1+4, 4
w1 = not(w1)
w2 = not(w2)
copy @r, @w1, 4
copy @r+4, @w2, 4
return r
END FUNCTION
FUNCTION ¤AND(byval quad v1, v2) as quad
dword w1
dword w2
quad r
copy @w1, @v1, 4
copy @w2, @v2, 4
w1 = (w1 and w2)
copy @r, @w1, 4
copy @w1, @v1+4, 4
copy @w2, @v2+4, 4
w1 = (w1 and w2)
copy @r+4, @w1, 4
return r
end function
FUNCTION ¤OR(byval int v1, v2) as int
return v1 or v2
end function
'FUNCTION ¤OR(byval quad v1, v2) as quad
' dword w1
' dword w2
' quad r
' copy @w1, @v1, 4
' copy @w2, @v2, 4
' w1 = (w1 or w2)
' copy @r, @w1, 4
' copy @w1, @v1+4, 4
' copy @w2, @v2+4, 4
' w1 = (w1 or w2)
' copy @r+4, @w1, 4
' return r
'end function
FUNCTION ¤IMP(byval quad v1, v2) as quad
if v1 then return -1
if v2 then return -1
end function
FUNCTION ¤EQV(byval quad v1, v2) as quad
if v1=0 then return 0
if v2=0 then return 0
return -1
end function
FUNCTION ¤MOD(quad v1, v2) as quad
return MOD(v1, v2)
end function
' END OF SYSTEM_OPERATORS.BIN
' CONTINUES (12) PLURIBASIC_PREPARE.BIN
#DEF HANDLE SYS
TYPE ¤SYSNMHDR
hwndFrom AS SYS
idFrom AS SYS
Code AS DWORD
END TYPE
class ¤SYSF
FUNCTION CONSTRUCTOR()
END FUNCTION
END CLASS
new ¤SYSF EXE()
' END OF PLURIBASIC_PREPARE.BIN
' STARTS PLURIBASIC_INIT.BIN
' This code is executed before anything else, if you want to do something before nything else, see PLURIBASIC_PREPARE
' END OF PLURIBASIC_INIT.BIN
' STARTS CALLBACKDATA.BIN
' END OF CALLBACKDATA.BIN
DECLARE FUNCTION PBMAIN() AS INT
' Initializes various things in the script.
FUNCTION PluriBASIC_Initialize() AS LONG
END FUNCTION
DECLARE FUNCTION PLAYSOUND LIB "WINMM.DLL" ALIAS "PlaySoundA" (P1 AS ASCIIZ, BYVAL P2 AS SYS, BYVAL P3 AS DWORD) AS INT
FUNCTION PBMAIN() AS INT
INT ¤RETVAL = 0
¤SYSERR Err
SYS h
h = GETMODULEHANDLE(0)
IF PLAYSOUND("WAV0001", h, 262148) THEN
print "Sound played"
ELSE
print "Could not play sound"
END IF
RETURN ¤RETVAL
END FUNCTION
PBMAIN() ' invoke entry point
Attached is the generated executable.
-
Brian, your binary works on my PC, after unblocking :)
-
Then i am clueless... I dont know why it doesnt produce any sounds in my PC... It just hangs there until the time my IDE allows it to tun elapses.
Thanks Charles... i will keep looking.
-
ResEdit shows a WAVE folder with WAV0001 resource, tagged Spanish (Mexico). Similar with the icon.
Sound.exe also executes with an empty console window.
-
Hello Charles,
Well, its good to know it works... but its driving me crazy that it doesnt work in MY pc. This is the
same script in PowerBASIC... somewhat. It compiles and plays the sound fine. Mesage box displays: "Sound played"
#COMPILE EXE
#RESOURCE WAVE, WAV0001, "executing.wav"
%SND_SYNC = &H0000
%SND_ASYNC = &H0001
%SND_RESOURCE = &H00040004
DECLARE FUNCTION PLAYSOUND LIB "WINMM.DLL" ALIAS "PlaySoundA" (azsSound AS ASCIIZ * 30, BYVAL hModule AS DWORD, BYVAL Opts AS DWORD) AS LONG
DECLARE FUNCTION GetModuleHandle LIB "Kernel32.dll" ALIAS "GetModuleHandleA" (lpModuleName AS DWORD) AS DWORD
FUNCTION PBMAIN() AS LONG
LOCAL sn AS ASCIIZ * 30
SN = "WAV0001"
LOCAL h AS DWORD
H = GetModuleHandle(BYVAL 0)
IF PlaySound(sn, h, %SND_RESOURCE OR %SND_SYNC) THEN
? "Sound played"
ELSE
? "Could not play sound"
END IF
END FUNCTION
Same code compiled for oxygen, and the program just executes idle, until it runs out of execution time. :(
Im utterly bewildered, i dont even know what to look for.
-
Perhaps... a manifest thing?
-
Charles,
How did you link a res file to the exe.
LinkRes2Exe does not work for me with WAVE.
James
-
There were no complaints here from LinkRes2Exe. but GORC issued a warning about WAVE as a user-defined type
A WAVE resource is a bit unusual anyway. Perhaps leave it for now and see if any other problems emerge with resources.
-
You are right Charles. I Should rest. I have been focusing in other things today.
Now is time for enjoying life. :) A nice dinner with what is left from my family.
I believe we may need to look in another direction regarding the RC compiler or
the RC linker.
When i open the generated executable in ResEdit, and then simply click SAVE...
the executable magically starts working. It plays sounds flawlessly. Must be some
kind of missalignment that ResEdit handles better.
By the Way, i am using Win7, what OS are you running? I have heard Win7 is
less forgiving.
Thanks for enduring me! cya later. :)
-
Charles... out of curiosity i opened in ResEdit the tetris clone because it had the same issue (no sound):
https://www.oxygenbasic.org/forum/index.php?topic=1910.msg20766;topicseen#msg20766
It says there is an 'invalid resource 6' (check picture). Freshly compiled, with no resources.
Once the resource is removed or the exe is resaved, the game plays the sounds perfectly fine. :o
-
Finally got it working. Now.. off to sleep!
-
Charles,
My OCD would not let go :)
The issue is WAVE does not have a resource id number associated with it so LinkRes2Exe cannot add it to the exe. I wanted to make sure it was not a gorc issue so I tried the latest (1.07) 64bit FreeBasic
Created using RadAsm3.
James
'==============================================================================
FreeBasic code:
#include "windows.bi"
#include "win\mmsystem.bi"
print "hello"
PlaySoundA("#100",GetModuleHandle(0),SND_RESOURCE OR SND_ASYNC)
print "any key to end"
sleep
resource file
#define IDC_WAV 100
IDC_WAV WAVE DISCARDABLE "executing.wav"
output:
gorc /r "playwav.rc"
GoRC.Exe Version 1.0.1.0 - Copyright Jeremy Gordon 1998-2013 - JG@JGnet.co.uk
Warning .......................
Line 3 of Resource Script (playwav.RC):-
A resource type was not recognised and assumed to be user-defined:-
WAVE
Output file: playwav.res
fbc -s console "playwav.bas" "playwav.res"
-
The issue is WAVE does not have a resource id number associated with it...
This is a working RC file (sound splays OK now), apparently WAVE can have an ID. Its just that, as Charles said, GoRc bites one character off of it...
xWAV0001 WAVE "C:\\Users\\Diamante\\Documents\\PluriBASIC\\Clean\\executing.wav"
Since i want WAV0001, i need to prepend a character... x.
-
I've removed the o2 resource section stub. So you will have clean binaries before adding resources.
All you need is the updated OxygenProgress.zip for oxygen.dll
I've also discovered that GORC compiles the symbol names correctly, therefore the 1st character is being snipped off by LinkRes2Exe. We have the source code and I will see if this can be fixed.
-
Charles, speaking of that... i dont know BCX, but if you convert LinkRes2Exe to Oxygen, i can implement it
in my oxygen caller and make one seamless call to perform all tasks at once.
-
Charles,
This will let us use LinkRes2Exe with imbedded wav files as RCDATA instead of WAVE.
James
use rtl64
use corewin
% SND_MEMORY = 4
DECLARE FUNCTION sndPlaySound LIB "WINMM.DLL" ALIAS "sndPlaySoundA" (azsSound AS ASCIIZ,BYVAL Opts AS DWORD)AS INT
DECLARE FUNCTION PBMAIN() AS INT
'==============================================================================
Function PlayRcSound (hInst as sys, SndName As sys) As Integer
Dim As sys hr,hg,id,lpSndData
id = 10
hr = FindResource (hInst, (char*)SndName, (char*)id)
If hr <> 0 Then
hg = LoadResource (hInst, hr)
If hg <> 0 Then
lpSndData = LockResource (hg)
If lpSndData <> 0 Then
sndPlaySound ((char*)lpSndData, SND_MEMORY)
return 1
Else
return 0
End If
Else
return 0
End If
Else
return 0
End If
Function = 0
End Function
'==============================================================================
Sub main()
If PlayRcSound(GetModuleHandle(0),100)
Print "Sound Played"
Else
Print "No Sound"
End If
End Sub
main()
'==============================================================================
Resource File
#define IDC_WAV 100
IDC_WAV RCDATA DISCARDABLE "executing.wav"
-
Charles,
I looked at LinkRes2Exe source and I don't think it's possible to fix.
It looks for the resource type id (manifest = 24, rcdata = 10 ...) and WAVE was never assigned an id by Microsoft that I could find.
James
-
Charles,
I'm a bit red faced. I recompiled LinkRes2Exe and it now does not abort and adds the WAVE but it will not play as you noted.
used bc9 and Pelles 8.0 64bit on translated code.
James