The MASM Forum

General => The Workshop => Windows API => Topic started by: jj2007 on November 20, 2023, 03:24:09 AM

Title: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on November 20, 2023, 03:24:09 AM
A simple template demonstrating how elegant and efficient Assembly can be

include \masm32\include\masm32rt.inc ; **** load standard Masm32 includes ****
.data ; initialised data section
wc WNDCLASSEX <WNDCLASSEX, 0, WndProc, 0, 0, 1, 2, 3, COLOR_BTNFACE+1, 0, txClass, 4>
txClass db "Masm32GUI", 0 ; class name, will be registered below
msg MSG <>
.code
WinMain:
  mov wc.hInstance, rv(GetModuleHandle, 0) ; rv ("return value") is a Masm32 macro
  mov wc.hIcon, rv(LoadIcon, eax, IDI_APPLICATION)
  mov wc.hIconSm, eax ; the rv macro returns results in eax
  invoke RegisterClassEx, addr wc ; the window class needs to be registered
  invoke CreateWindowEx, WS_EX_ACCEPTFILES, wc.lpszClassName, chr$("Hello World"), ; title
     WS_OVERLAPPEDWINDOW or WS_VISIBLE, 120, 120, 300, 120, ; style, x, y, width, height
     NULL, rv(LoadMenu, wc.hInstance, 100), wc.hInstance, NULL
  .While 1
inc rv(GetMessage, addr msg, 0, 0, 0)
shr eax, 1
.Break .if Zero? ; 0 (OK) or -1 (error)
invoke TranslateMessage, addr msg
invoke DispatchMessage, addr msg
  .Endw
  invoke ExitProcess, msg.wParam
WndProc proc hWnd, uMsg, wParam:WPARAM, lParam:LPARAM
  .if uMsg==WM_DESTROY
invoke PostQuitMessage, NULL
  .endif
  invoke DefWindowProc, hWnd, uMsg, wParam, lParam ; default processing
  ret
WndProc endp
end WinMain
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: NoCforMe on November 20, 2023, 08:32:29 AM
Impressive. But didn't you leave out the lines in the resource file? That menu didn't just appear out of thin air ...
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on November 20, 2023, 09:48:08 AM
Well, well... strictly speaking that isn't code :cool:
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: TimoVJL on November 20, 2023, 09:27:57 PM
From WinMin examples:
#define WIN32_LEAN_AND_MEAN
#include <windows.h>
#include <shellapi.h>

#pragma comment(lib, "shell32.lib")

static LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
static void OnDropFiles(HWND hWnd, HDROP hDrop);

static TCHAR *szAppName = TEXT("WinFrame");
static TCHAR *szFrameClass = TEXT("cWinFrame");
static WNDCLASSEX wcx = {sizeof(WNDCLASSEX), CS_HREDRAW | CS_VREDRAW, 0, 0, 0,
0, 0, 0, (HBRUSH)COLOR_3DSHADOW, 0, 0, 0};
HWND g_hFrame;
HANDLE g_hInst;

void __cdecl WinMainCRTStartup(void)
{
ExitProcess(WinMain(GetModuleHandle(NULL),0,0,SW_SHOW));
}

int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow)
{
MSG msg;
wcx.hInstance = hInstance;
wcx.lpfnWndProc = WndProc;
wcx.lpszClassName = szFrameClass;
wcx.hCursor = LoadCursor(NULL, IDC_ARROW);

if (!RegisterClassEx(&wcx))
return 0;
g_hInst = hInstance;

g_hFrame = CreateWindowEx(0, szFrameClass, szAppName,
WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT,
CW_USEDEFAULT, CW_USEDEFAULT,
NULL, NULL, hInstance, NULL);
if(!g_hFrame) return 0;
ShowWindow(g_hFrame, nCmdShow);
DragAcceptFiles(g_hFrame, TRUE);
while(GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
DragAcceptFiles(g_hFrame, FALSE);
return msg.wParam;
}

static LRESULT CALLBACK WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch(uMsg) {
// case WM_PAINT:
// case WM_CREATE:
case WM_DROPFILES: OnDropFiles(hWnd, (HDROP)wParam); return 0;
case WM_DESTROY:
PostQuitMessage(0);
return(0);
}
return DefWindowProc(hWnd, uMsg, wParam, lParam);
}

static void OnDropFiles(HWND hWnd, HDROP hDrop)
{
TCHAR szTmp[260];
int nCnt = DragQueryFile(hDrop, -1, 0, 0);
if (nCnt) {
int nLen = DragQueryFile(hDrop, 0, szTmp, sizeof(szTmp));
MessageBox(hWnd, szTmp, "File:", MB_OK);
}
}
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on November 21, 2023, 11:42:36 PM
Quote from: TimoVJL on November 20, 2023, 09:27:57 PMFrom WinMin examples:

72 lines of C code, nice :thumbsup:

With the right tools and libraries, one can do fantastic things:
GuiParas equ "Hello World", w200, h400, b LiteBlueGreen, icon Printer    ; width+height, background colour
GuiMenu equ @File, &Open, &Save, &Print, -, E&xit, @Edit, Undo, Copy, Paste
include \masm32\MasmBasic\Res\MbGui.asm
Event DropFiles    ; loads the Files$() array; use wRec$() to convert Utf8 file names to true Unicode
 wMsgBox 0, wRec$(Join$(Files$())), "Files dropped:", MB_OK or MB_TOPMOST
 StoreUtf8 "~tmpDropped.txt", Files$()    ; write dropped filenames to a text file in Utf8 format
GuiEnd

Seven lines of code, purest Assembly (plus, it writes the filenames to disk, see attachment...) ;-)
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: TimoVJL on November 22, 2023, 12:48:57 AM
Quote from: jj2007 on November 21, 2023, 11:42:36 PM...With the right tools and libraries, one can do fantastic things:
with that principle, just one line is enough ?
Those who have read Dr. Dobbs Magazine, knowns, what i am talking about  :smiley:
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: NoCforMe on November 22, 2023, 08:53:08 AM
Yeah, I think this whole "look what I can do in n lines of code!" business is far too overrated. After all, sometime in the future, after neural-machine links are perfected** and all, our code will reduced to this:
DWIM
which of course is the instruction for "do what I mean".

I especially think using macros in this way is cheating, as they compound their constituent "lines of code" (instructions) into one deceptive-looking "instruction":
  mov wc.hInstance, rv(GetModuleHandle, 0) ; rv ("return value") is a Masm32 macro
should properly be expanded into what it really is, which is
  invoke GetModuleHandle, 0
  mov wc.hInstance, eax
(and if you're a purist, which I am not, you could even argue that "invoke" should be expanded into its push-push-call constituents)

** I sincerely hope I'm not still alive when that happens.
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on November 22, 2023, 09:59:38 AM
Quote from: NoCforMe on November 22, 2023, 08:53:08 AM(and if you're a purist, which I am not, you could even argue that "invoke" should be expanded into its push-push-call constituents)

That's exactly the point: where to draw the line? I find it silly to use push-push-call for CreateFont - it's so easy to count 13 instead of 14 args, and thus run into real problems because a stack imbalance often gets not noticed because of the stack frame.

So I use invoke CreateFont, ... 14 args ..., and the assembler will tell me if I counted wrong. Only that invoke is severely broken for the Masm64 SDK, it will not tell you that you missed an arg, or added one too many.

Of course, instead of using Recall (https://www.jj2007.eu/MasmBasicQuickReference.htm#Mb1172), I could paste a dozen lines of code to simulate get text from a file and put it into an array. Will it make my code more readable? Will it make me happier because I demonstrate that I can code pure assembly copy & paste?

I have over 4,000 sources in my Masm32 folder. I refuse to use "pure" Assembly.
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on February 21, 2024, 11:09:08 PM
Update: attached a rather bloated 37 lines window template, including a Gdi32 demo - enjoy :thup:
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: daydreamer on February 22, 2024, 01:58:11 AM
Wonder if guga can beat that with oneliner, with his multiple mnemonics on one line caps assembler ?
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: NoCforMe on March 12, 2024, 09:57:25 AM
An example (https://tromp.github.io/tetris.html):
long h[4];t(){h[3]-=h[3]/3000;setitimer(0,h,0);}c,d,l,v[]={(int)t,0,2},w,s,I,K
=0,i=276,j,k,q[276],Q[276],*n=q,*m,x=17,f[]={7,-13,-12,1,8,-11,-12,-1,9,-1,1,
12,3,-13,-12,-1,12,-1,11,1,15,-1,13,1,18,-1,1,2,0,-12,-1,11,1,-12,1,13,10,-12,
1,12,11,-12,-1,1,2,-12,-1,12,13,-12,12,13,14,-11,-1,1,4,-13,-12,12,16,-11,-12,
12,17,-13,1,-1,5,-12,12,11,6,-12,12,24};u(){for(i=11;++i<264;)if((k=q[i])-Q[i]
){Q[i]=k;if(i-++I||i%12<1)printf("\033[%d;%dH",(I=i)/12,i%12*2+28);printf(
"\033[%dm  "+(K-k?0:5),k);K=k;}Q[263]=c=getchar();}G(b){for(i=4;i--;)if(q[i?b+
n[i]:b])return 0;return 1;}g(b){for(i=4;i--;q[i?x+n[i]:x]=b);}main(C,V,a)char*
*V,*a;{h[3]=1000000/(l=C>1?atoi(V[1]):2);for(a=C>2?V[2]:"jkl pq";i;i--)*n++=i<
25||i%12<2?7:0;srand(getpid());system("stty cbreak -echo stop u");sigvec(14,v,
0);t();puts("\033[H\033[J");for(n=f+rand()%7*4;;g(7),u(),g(0)){if(c<0){if(G(x+
12))x+=12;else{g(7);++w;for(j=0;j<252;j=12*(j/12+1))for(;q[++j];)if(j%12==10){
for(;j%12;q[j--]=0);u();for(;--j;q[j+12]=q[j]);u();}n=f+rand()%7*4;G(x=17)||(c
=a[5]);}}if(c==*a)G(--x)||++x;if(c==a[1])n=f+4**(m=n),G(x)||(n=m);if(c==a[2])G
(++x)||--x;if(c==a[3])for(;G(x+12);++w)x+=12;if(c==a[4]||c==a[5]){s=sigblock(
8192);printf("\033[H\033[J\033[0m%d\n",w);if(c==a[5])break;for(j=264;j--;Q[j]=
0);while(getchar()-a[4]);puts("\033[H\033[J\033[7m");sigsetmask(s);}}d=popen(
"stty -cbreak echo stop \023;sort -mnr -o HI - HI;cat HI","w");fprintf(d,
"%4d from level %1d by %s\n",w,l,getlogin());pclose(d);}

Ugly as shit.
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 12, 2024, 08:31:57 PM
Quote from: jj2007 on February 21, 2024, 11:09:08 PMUpdate: attached a rather bloated 37 lines window template, including a Gdi32 demo - enjoy :thup:


how about refactor these 2 lines

invoke BeginPaint, hWnd, addr ps
   mov hDC, eax


into singleliner?

using rv() marco?
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 12, 2024, 10:19:03 PM
Also the msg loop is a bit unusual...

usually I'm using this one:

.while TRUE
invoke GetMessage,addr msg,NULL,0,0
  .BREAK .if !eax
invoke TranslateMessage,addr msg
invoke DispatchMessage,addr msg
.endw

Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 13, 2024, 12:00:28 AM
Quote from: greenozon on March 12, 2024, 08:31:57 PMinto singleliner?

using rv() marco?

Your wish is my command, see attachment :biggrin:

Quote from: greenozon on March 12, 2024, 10:19:03 PMAlso the msg loop is a bit unusual...

.BREAK .if !eax
What happens with your code if GetMessage returns -1?

QuoteGetMessage function (https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-getmessage)
Return value
Type: BOOL

If the function retrieves a message other than WM_QUIT, the return value is nonzero.

If the function retrieves the WM_QUIT message, the return value is zero.

If there is an error, the return value is -1. For example, the function fails if hWnd is an invalid window handle or lpMsg is an invalid pointer. To get extended error information, call GetLastError.

Compare to the code in Windows36lines:
inc rv(GetMessage, addr msg, 0, 0, 0)
shr eax, 1
.Break .if Zero? ; 0 (OK) or -1 (error)
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 13, 2024, 02:58:29 AM
jj, nice!!

going back to
>>If there is an error, the return value is -1. For example, the function fails if hWnd is an invalid window handle or lpMsg is an invalid pointer. To get extended error information, call GetLastError.

tell me, from the bottom of your heart,
what is the % of probability in your specific piece of (great) SW that this error might happen?


Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 13, 2024, 03:02:45 AM
another idea/question to get -1 line ...

how about this:

it is possible to make 1 line instead of these 2?

wc   WNDCLASSEX <WNDCLASSEX, 0, WndProc, 0, 0, 1, 2, 3, COLOR_BTNFACE+1, 0, txClass, 4>
txClass   db "Masm32GUI", 0         ; class name, will be registered below


focus on txClass
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 13, 2024, 04:07:19 AM
Quote from: greenozon on March 13, 2024, 02:58:29 AMtell me, from the bottom of your heart,
what is the % of probability

Unless you have an extremely valid reason to ignore the documentation, don't do it.

Quote from: greenozon on March 13, 2024, 03:02:45 AMit is possible to make 1 line instead of these 2?

The txClass string is being used twice. You might use chr$(..) instead and save one line, but in contrast to MasmBasic's Chr$(..) (https://www.jj2007.eu/MasmBasicQuickReference.htm#Mb1143), the Masm32 SDK chr$() generates two entries for identical strings - therefore I would hate that :cool:
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: sinsi on March 13, 2024, 05:48:56 AM
Quote from: sinsi on May 21, 2014, 03:01:57 PMWhen will GetMessage return -1? (http://blogs.msdn.com/b/oldnewthing/archive/2013/03/22/10404367.aspx)

Quote from: Raymond ChenBut don't worry, the standard message pump is safe. If your parameters are exactly
•a valid pointer to a valid MSG structure,
•a null window handle,
•no starting message range filter,
•no ending message range filter,

then GetMessage will not fail with -1.
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 13, 2024, 05:53:53 AM
Quote from: jj2007 on March 13, 2024, 04:07:19 AMThe txClass string is being used twice.

I've read the .asm code thrice  and... I can't find the used twice case!?
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 13, 2024, 07:02:34 AM
Check the WNDCLASSEX structure:

Quotewc    WNDCLASSEX <WNDCLASSEX, 0, WndProc, 0, 0, 1, 2, 3, COLOR_BTNFACE+1, 0, txClass, 4>
txClass    db "Masm32GUI", 0            ; class name
...
  invoke CreateWindowEx, WS_EX_ACCEPTFILES, wc.lpszClassName, chr$("Hi there") ...

However, you are right that it can be a one line shorter, see attachment :cool:

Quote from: sinsi on March 13, 2024, 05:48:56 AMIf your parameters are exactly
Exactly, Sinsi ;-)

(if not, it's called a bug, and if you don't play the shr eax, 1 trick, the program will most likely become an invisible zombie that must be manually killed in Task Manager)
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: sinsi on March 13, 2024, 07:32:29 AM
Two things to cause an error, but you break for the error and a quit message. To be properly paranoid your code should jump to an error handler if the return code is -1.

He also warns against using any sort of filter, as another window (e.g. OLE, CBT) might use your window procedure, and since GetMessage is blocking it might have messages piling up that don't match your filter but other code relies on them being processed.
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 13, 2024, 10:08:28 AM
Quote from: sinsi on March 13, 2024, 07:32:29 AMyou break for the error and a quit message.

Yep, that's the intention. Here is a loop that is foolproof:

  invoke CreateWindowEx, WS_EX_ACCEPTFILES, wc.lpszClassName, chr$("Hi there"), ; class, title
     WS_OVERLAPPEDWINDOW or WS_VISIBLE, 120, 120, 240, 180, ; style, x, y, width, height
     NULL, rv(LoadMenu, wc.hInstance, 100), wc.hInstance, NULL
  .While eax
inc rv(GetMessage, addr msg, 0, 0, 0)
shr eax, 1
.Break .if Zero? ; 0 (OK) or -1 (error)
invoke TranslateMessage, addr msg
invoke DispatchMessage, addr msg
or eax, -1
  .Endw
  invoke ExitProcess, msg.wParam

If there is an error when creating the window, the loop will not be entered at all. The or eax, -1 makes sure that the loop keeps going until WM_QUIT or an error occur. Back to 36 lines, unfortunately, but it's worth it - I checked, an error in CreateWindowEx will indeed create a zombie...
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: TimoVJL on March 13, 2024, 10:22:15 PM
Sure, on purpose.
WS_EX_ACCEPTFILES
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 13, 2024, 11:10:34 PM
Quote from: TimoVJL on March 13, 2024, 10:22:15 PMSure, on purpose.
WS_EX_ACCEPTFILES

Indeed, using this style would require an appropriate handler, which is a bit beyond this simple template.

include \masm32\MasmBasic\Res\MbGui.asm
  GuiControl MyEdit, "RichEdit", wCL$()
  invoke SetWindowPos, hGui, HWND_TOPMOST, eax, eax, eax, eax, SWP_NOMOVE or SWP_NOSIZE

Event DropFiles  ; load a dropped text or rtf file into the RichEdit control
  SetWin$ hMyEdit=FileRead$(Files$(0))
GuiEnd

You can test it with the attached Ansi, Utf16 and RTF files ;-)
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 13, 2024, 11:42:57 PM
the last DropFiles.exe looks interesting indeed!
it could handle .rtf (wow!)
but... the emoji's support is very limited, it was able to render a couple out of hundreds available...
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 14, 2024, 05:51:45 AM
Quote from: greenozon on March 13, 2024, 11:42:57 PMthe emoji's support is very limited, it was able to render a couple out of hundreds

Can you post some examples that didn't work?
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 14, 2024, 08:13:20 AM
It's easy
goto https://emojipedia.org/flags
then copy whatever you like, eg:

🏁
🚩
🎌
🏴
🏳�
🏳��🌈
🏳��⚧️
🏴�☠️
🇺🇳

and paste into the DropFiles.exe editor pane

or whatever emojis you like eg
😺😺😺
😱
😨
😰
😥
😓
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 14, 2024, 08:44:21 AM
No problem for SimpleEditor :cool:

(https://i.postimg.cc/kB2QgC9S/Emojis.png) (https://postimg.cc/kB2QgC9S)
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: sinsi on March 14, 2024, 11:17:30 AM
Easy emojis in Windows 11

Windows+.
😁😊🤣😂❤️👍

Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 14, 2024, 12:44:04 PM
Quote from: sinsi on March 14, 2024, 11:17:30 AM😁😊🤣😂❤️👍

Fine for Win10 & SimpleEditor (https://masm32.com/board/index.php?topic=11759.0) - if saved as rtf, see attachment :thumbsup:
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: sinsi on March 14, 2024, 01:01:20 PM
Notepad and Notepad++ seem to save as normal text.
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: jj2007 on March 14, 2024, 09:41:51 PM
Hi Sinsi,

This is weird: what you sent is Utf8 code without a Utf8 BOM:

F0 9F 98 8A  F0 9F A4 A3  F0 9F 98 82

So apparently, Notepad uses an autodetect mechanism that recognises these Utf8 codepoints. Interesting.

SimpleEditor uses a BOM, but when you build a project, it exports the code to Res\Tmp_file.asm in plain text format without BOM.

There it is (https://en.wikipedia.org/wiki/UTF-8):
QuoteWindows Notepad, in all currently supported versions of Windows, defaults to writing UTF-8 without a BOM (a change from the outdated/unsupported Windows 7 Notepad)

This reminds me of our lovely discussions with Utf8 hater Nidud ;-)

Since I am an expert for creating confusion, here is what happens when you write in your native codepage and then try to add emojis:

QuoteThe file contains Unicode characters that could be lost if the file were saved as a text file in ANSI characters. To preserve Unicode characters choose Undo, then select one of the Unicode options in the Encoding drop-down list. Continue?

Utf8 rocks, forget the old codepages :cool:

(https://i.postimg.cc/NydjDG6r/Not-Utf8-NP.png) (https://postimg.cc/NydjDG6r)

P.S., the page you are reading right now, btw, has a UTF8 BOM:
<meta charset="UTF-8">(https://upload.wikimedia.org/wikipedia/commons/9/97/Unicode_Web_growth.svg)
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: NoCforMe on March 15, 2024, 07:51:28 AM
Quote from: greenozon on March 14, 2024, 08:13:20 AMIt's easy
goto https://emojipedia.org/flags
then copy whatever you like, eg:

🏁
🚩
🎌
🏴
🏳�
🏳��🌈
🏳��⚧️
🏴�☠️
🇺🇳

and paste into the DropFiles.exe editor pane

or whatever emojis you like eg
😺😺😺
😱
😨
😰
😥
😓

Heh; forget about JJ's application for the moment: a lot of those characters don't even render correctly in my web browser (Opera 95.0.4635.90, Windows 7 64 Pro):
(https://i.postimg.cc/GTy8WmQ0/Emojis.jpg) (https://postimg.cc/GTy8WmQ0)
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 16, 2024, 04:02:19 AM
Try smth more recent
eg Supermium 122
Vivaldi 6.6
etc
!
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: NoCforMe on March 16, 2024, 06:48:33 AM
Sorry, not about to change my web browser!
Title: Re: A window in 30 lines of code, using the Masm32 SDK
Post by: greenozon on March 16, 2024, 07:41:06 AM
the you are under super high risk rihgtnow!
if you still using geriatric build 109