A styleguide and resource list for the Microsoft Assembler (MASM)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Taylor Bockman 0e599cb357
Add Alignment Section
7 years ago
LICENSE Initial commit 7 years ago
README.md Add Alignment Section 7 years ago

README.md

MASM Styleguide and Resource List

The Microsoft Assembler (MASM) is a popular assembler for Windows platforms. Unsurprisingly, there has been very few notable projects in MASM in the last decade. Small, fast code just isn't as important as it used to be. Still, it is a very interesting exercise to learn MASM and use it. It's difficult to get binary sizes smaller than you can with raw assembly. Additionally, if you're in the world of reverse engineering, programming in assembly keeps your languages consistent.

Table of Contents

  1. Getting Started
  2. The Assembler
  3. IDEs
  4. VIM Plugins
  5. Visual Studio Code Extensions
  6. Tutorials
  7. Styleguide
  8. Tabs vs. Spaces
  9. Line Length
  10. Commenting
  11. Include Ordering
  12. Case Sensitivity
  13. Section Names
  14. Variable Names
  15. Functions
  16. Function Prototypes
  17. Section
  18. Type Declarations
  19. Alignment
  20. Project Organization

Getting Started

You're going to want to get some development tools. Of course, it goes without saying you're going to need some flavor of Windows to run this stuff. Once you've acquired Windows, you will need the assembler and an IDE (optionally).

The Assembler

The most popular variant of MASM is MASM32. It's more or less an all in one package for coding assembly programs on Windows. One caveat - you'll need Windows 2000 or greater. This guide will focus on MASM32 primarily. Suggestions here should be easily ported to other variants.

IDEs

Thanks to the popularity of MASM, there are a few IDE choices if you choose to go that route. They are helpful, in that they handle assembling and linking for you. However, it's honestly something you can handle with a script.

WinAsm Studio

WinAsm Studio is arguably the most popular IDE for writing assembly code on Windows (as far as I've known). As it stands, I've had difficulty creating a new account on their forum to download the IDE. Unfortunately, they "account wall" the download so if you can't make an account you won't get the download. It's quite silly.

VisualMASM

VisualMASM is another popular IDE. It's tailored to MASM and has the look and feel of a modern IDE. The most frustrating aspect is that there's no option to create a blank project. So if you're using a template, you're going to need to delete the base code that comes with the project to get anywhere. I have an issue on the VisualMASM github to address this.

VIM Plugins

VIM has pretty good support for assembly programming out of the box. I couldn't find any plugins that made programming any easier.

Visual Studio Code Extensions

x86 and x86_64 Assembly seems to be the most popular extension for Visual Studio Code when it comes to developing in assembly.

Tutorials

The most popular set of tutorials for MASM32 is Iczelion's Tutorial Series. There is a .CHM version of these tutorials available on the WinAsm website.

Styleguide

The styleguide is a work in progress. As I find more pretty ways to write assembly code I will put them here. I've gathered over my time writing MASM code that every developer does things differently. This makes it incredibly frustrating to try and read another developer's code.

PS: It would be really cool if someone wrote a linter that checked for these things. Maybe I'll get around to it one of these days.

Tabs vs. Spaces

TODO

Line Length

Prefer 120 characters. MASM has a hard limit of 255 characters per line according to this post on MASM Forum. Many times prototypes and function invocations can have very long line lengths. For these, use \ to split your code across multiple lines.

Commenting

TODO

Include Ordering

As a matter of style, prefer to put all .inc files together and all .lib files together. Sort them alphabetically, and separate them by a single blank line.

Example:

include \masm32\include\kernel32.inc
include \masm32\include\user32.inc
include \masm32\include\windows.inc

includelib \masm32\include\kernel32.lib
includelib \masm32\include\user32.lib

Case Sensitivity

Always use the option casemap:none option. This prevents the assembler from thinking MyFunction is the same as myfunction, which will save you a lot of headaches.

Example:

.386
.MODEL flat, stdcall
option casemap: none
include \masm32\include\windows.inc
include \masm32\include\kernel32.inc

includelib \masm32\lib\kernel32.lib

.DATA
.CODE
  start:
    invoke ExitProcess, 0
  end start

Section Names

Prefer uppercase section names. This will help differentiate them from the rest of the code at a quick glance.

Example:

.386
.MODEL flat, stdcall
option casemap: none

include \masm32\include\windows.inc
include \masm32\include\kernel32.inc

includelib \masm32\lib\kernel32.lib

.DATA
.CODE
  start:
    invoke ExitProcess, 0
  end start

Variable Names

TODO - Once this is clarified be sure to update any variable names in every section to reflect this change.

Functions

When writing functions, indent the body of the function so it is easily differentiated from other code levels.

Example:

   WinMain proc hInst: HINSTANCE, CmdLine: LPSTR, CmdShow: DWORD
    ; Code goes here...

    Ret
  WinMain EndP

Additionally, place your functions towards the bottom of your code section. This keeps your actual code uncluttered, and all of your functions in one easy to find place.

Function Prototypes

In order to allow the use of functions at the bottom of your code you need to declare function prototypes above the invoking code.

Place prototypes after all includes, and leave two empty lines between the includes and prototypes.

Example:

include \masm32\include\kernel32.inc
include \masm32\include\user32.inc
include \masm32\include\windows.inc

includelib \masm32\include\kernel32.lib
includelib \masm32\include\user32.lib


WinMain proto :DWORD, :DWORD, :DWORD, :DWORD

Function Arguments

Whether you are calling a function or writing an implementation, make sure to put a single space after each comma delimited argument.

Example:

  WinMain proc hInst: HINSTANCE, CmdLine: LPSTR, CmdShow: DWORD

Sections

In order to improve readability, indent the code in each section one level to differentiate it from the section name. This makes it so, when combined with all uppercase section names, It is easier to distinguish where you are in the code.

Example:

.CODE
  start:
    invoke ExitProcess, 0
  end start

Type Declarations

Put one space after the : in an argument's type declaration in order to improve readability. This goes the same for local variable type declarations.

Examples:

  WinMain proc hInst: HINSTANCE, CmdLine: LPSTR, CmdShow: DWORD
    LOCAL wc: WNDCLASSEX
    ; ...

Alignment

Prefer to keep type declarations and instantiations aligned.

Example:

.DATA 
  ClassName   db "WinClass", 0
  AppName     db "Window", 0

.DATA?
  hInstance   HINSTANCE ?
  CommandLine LPSTR     ? 

Project Organization

TODO