Sunday, October 10, 2010

Buzzword of the year - the cloud

Now everyone does programming for 'the cloud'. That is the ultimate place to be with your 'app' (another buzzword) - go visit the cloud. I was once lucky enough to assist to a salesman praise of the cloud. It's horrific - I love it. The cloud has magic properties - you can do anything, achieve everything, f*** up nothing. Because, well, it's the cloud.

Now let's get things straight. The cloud doesn't exist. It's a bunch of computers put together. Google does that, Microsoft does that, Amazon does that, everyone does that. No magic - a bunch of computers put together. No superpowers granted - no super-technology. This was done since a lot of time. Now it's only a bit more 'public'. Around 1940s some dudes were talking about providing 'computer power' to everyone around an entire city and, why not, the world. 1940, no clouds, only dreamers. Now cloud, yet no computing power for everyone, just bullshit. Just some means to collect your data.

And while your information runs away from you  (because, well, it's in the cloud, why should I be more than an ignorant moron when I have 'the cloud') we sit back, relax and enjoy the new buzzword. Because everyone will take some more time from themselves, and the cloud will know them better than anyone. How does that feel? a bunch of machines know you better than you know yourself? But enough cyberpunk.

Anyway, fear the salesmen that throw the cloud at you. It's not a new concept - it's merely a name for the same things that happened since a lot of time.

PS: This post is now 'in the cloud'. It was once in social media. once, it was web 2.0. Once it was HTML. Once it was AJAX. Once it was on the net. Once it was online. My blog just can't settle, you know?

Monday, September 13, 2010

Wednesday, September 8, 2010

Something worth taking your time for

Since I complained for the non-Agile Agile methods, some thoughts that I feel correspond quite well with what I feel on this can be read here. Also, I want to point out this incredibly nice video I seen from that link (and btw, link was received from the unfoundable-on-the-net bolthead).

In the next episode, hopefully I'll get back to technical stuff.

Wednesday, September 1, 2010

The world of scribblers and slaves

I was always in opposition of software development processes, because I always felt that the processes are meant to diminish the skill of individuals creating software, adding an overhead that creates no value.

Seems, I was right. Indeed, I always preferred ad-hoc organization, direct responsibility for code and for the results of what you do. I overlooked intentionally the politics involved in software development - all the managerial chess-playing that makes the product go live and live instead of being a simple storage issue in the code repository. It might seem like a huge thing, but it's really not - it's people doing their job. The software processes were meant to make the people that usually don't get along to get along. Unfortunately, software writing people are not the ones that have a say in this, so software processes are fitting better the managerial people rather than the people actually doing the work, the software people. Ad-hoc processes usually mean that the software writing guys doing things as they know best. Established processes is software writing guys doing things the marketing way. And it doesn't work.


The current disappointment I have regarding the processes concerns SCRUM, but I'm pretty sure that it's not just SCRUM... it's all of them. All the software processes are meant to transform the programmers into working bees, working bees that produce the same quantity of honey month after month, even if it's winter time and no flowers are around. Useful, predictable slaves, that, like in the fourth grade math problems, 'dig 20 meters of trench in 2 day, therefore 100 meters in 10 days'. Everything is measured, the creativity is measured. But there's a catch.

Imagine SCRUM in poetry. People gathering around, looking at how many verses you finished today, which one will you write tomorrow, what impediments you have in writing your poetry. But you can't really calculate how much poetic spirit is in a poem, can you? So there's a solution from the managerial team. What we used to call 'self-critic' - something that reminds me so badly of the communist time - but now it's called "inspect and adapt" or, even better, "retrospective". We do it, because that gives us freedom. The freedom to continue with this world of scribblers and slaves. The scribblers accounting for how many poetry points you scored today, why your score wasn't good enough, and how can you improve. How fast can you make old ladies cry, why didn't you do it faster or in better shape.

And all this time, you'll wonder why your poetry lacks soul, biting from a tasteless tomato, in a sucky country, preparing to leave to the place of scribblers and their slaves.

Wednesday, July 7, 2010

C guideline: Extern declarations

One thought, as I was browsing through tons of bad code I have here at hand:

The C programming language allows you to write extern declarations quite everywhere. In the C file, in the function, even inside your 'for in for in for'. This usually tends to show the lack of structure and understanding of what should stay where. Externals, like globals, are bad code, so you might want to rethink the whole code you're working on.

If you still want to use it, probably the only place where you should be allowed to use them is in header files. Adding extern declarations in code may add unforeseeable linking problems, since the declaration of the original item might lie in a different module. Logically, the extern declaration is part of the interface of a module, thus it should lie in the header file that you include to access the module functionality.

If you want to get rid of this, you can simply add for the globals that you want to access setters and getters, depending on what you need. It would be the cleaner version anyway, and it will help you refactor things very fast and easy.

So as a guideline: don't use extern declarations in your C file. Move it in the header file or remove the need of the extern declaration altogether.

Tuesday, June 15, 2010

The boy scout rule

When you leave the camp, leave it a bit better than you found it . From this presentation (which I found interesting, but very annoying at times, and the beginning is simply too silly)(via Victor Hurdugaci). Which applied to code is:

Leave the code you fix a bit better than you found it. Fix one small thing or two. Rearrange some code. make it work a bit better.

I think it's an incredibly good piece of advice - bugs usually creep in badly written code - and by fixing badly written code whenever you see it you might fix a bug before it appears.

Monday, June 14, 2010

TDD stupid question

Ok, repost, too many mistakes.

I was watching a video with a guy very excited about TDD (a big mdeah from my side) - but mostly by the most inane part of it, which is 'pair programming'.

Why is that a good idea? Doesn't it sound like a "two stupid people put together might make as much as a good programmer, hopefully" thing?

Pair programming sucks big time - two people cannot write the same line of code, get the heck over it.

Thursday, May 6, 2010

C# for C programmers. Episode 1: Delegates

I was browsing the net looking for cute stuff, and reached this place. And I think that for the first time I understood what the buzz word "Delegate" means.

It's a pointer to a function, only in a nicer shell.

So instead of having:
typedef void (*pFunc) (const char *param1);
We will have a much nicer:
delegate void pFunc (string param1);

A cool thing that totally justifies the update from native code to almost native (but interpreted) code.

So next time someone bothers you with cool features that should make you swarm to .NET, you can find out what (at least one) of those buzz words mean.

For the real documentation of the delegate thingie, see MSDN. (via Bolthead)

Sunday, February 21, 2010

Python: my first 2 days

Let's start with what I really don't like: Invisible syntax sucks. Big time.

When your application complains about some spaces in your indentation, you know that you've done something wrong. So you load the source back in the editor, but you don't get it. It looks perfect. But it's not. This is because tabs are different from spaces, and spaces and tabs are a very very very different thing. So Python says.

There is nothing cool about this particular feature, except the enforcing of 'good looking code', and I find it very very confusing.


On the plus side, throwing together pieces of code that would simply be run is quite cool, not to mention the ease with which you get to do certain operations when compared with C/C++. On the other hand, the ease of use is actually reflected in the run times and probably beginner users like me will always make sluggish applications.

Another thing that I don't like so much is the fact that you don't have error checking on the run trails that you don't get to touch. So for example if you write a function and you have somewhere a misspelling, the error might pop up to you only in the rare case you might hit that line/trail. This is where a compilation stage becomes really really helpful, in my opinion, because quite some time is wasted on getting your code tested for the basic requirements (like not mistaking setGeometry for setgeometry).

The second part of my explorations with PyQt is the Qt4 framework(s), quite different conceptually from what I used to know from the Qt3 that I used so much some time ago. Better? Arguable, but my instincts were trained to use the Qt3 way of thinking, so I am not drawing any conclusions.

In the mean time I am really happy to say that it's pretty fun to use so far. If performance is not necessarily your goal, Python might be the language of choice.

Sunday, January 31, 2010

HTML5 video playback: Ogg/Theora or MPEG4/H264 (and a GIF story)

Do you remember GIF? You see some pictures on the web from time to time, rarely. The GIF format was pretty good for its time - it had lossless image compression (pretty good I might say), it had animations, transparency - pretty much anything you might want from an image on the web. And it was widely used in the '90s websites. And then, came Unisys, who remembered that, well, it had patents on the GIF compression method, and they wanted some money from those who used it. Just remember this campaign: Burn all gifs.

Their patent expired in 2003/2004, but some threats still linger. And nobody will use GIFs now, because, well, they were bested by the PNG format (and the lesser used MNG format). And even if I am old enough to remember the story, and yet I'm only 30, nobody seems to remember this story.


And the story happens again.This time it's not images, it's videos. And it's not like we have to make up an alternative FAST, like we did with the PNG format. The alternative is here, it's called Theora. Yes. I'm talking about the H264 standard used nowadays by Google or Vimeo to stream video over the internet, and that is now being pushed as 'HTML5' (although the HTML5 didn't specify the precise codec for the video playback). Ok, so what's the problem (aside from the fact that Mozilla won't support HTML5 H264 rendering?) Well, the problem is that soon enough, if you'll have a site and you want show some videos on it, you might be forced to pay for the use of the codec. Probably not what you have in mind.

A clearer article about this issue you can read from here; please also sign the petition addressed to Google to support in Ogg/Theora in youtube.

Sunday, January 17, 2010

Linked list complexity redivivus

Because fast doesn't always mean simpler, sometimes we have to get very very dirty.

Yesterday we left the problem of the memory allocation in suspense. Today we will try to get deeper in the muddy waters of O(1) allocation. And then, some unicorns.

Since we know that we cannot dream of constant time memory allocation (unless unicorns are involved) the best idea would be to preallocate the items in a single batch. So, let's say that we need maximum 10.000 elements, we allocate space for those maximum 10.000 elements from the beginning with a single malloc.

Good, but how do we actually take from that memory pool elements in constant time? Well... Theory has it that we could use a linked list for that ;)

Nobody tells us that the creation time has to be small, right? So when we allocate the space for those elements we can link together those elements, right?

node *mem_pool_create (int maxelem)
{
   node *head = malloc (sizeof (node)*maxelem);
   int i;
   if (!head) return NULL;
   for (i=0; i= 1 */

      head[i]->next = head+(i+1);
   head[maxelem-1]->next = NULL;
   return head;
}
/* bingo! Constant time! */
node *mem_pool_extract (node **pool_head)
{
   node *ptr;
   if (!pool_head || !*pool_head) return NULL;
   ptr = *pool_head;
   *pool_head = ptr->next;
   return ptr;
}
/* by re-adding the node in front we make sure the nodes
 * won't get too scattered in memory - and we might take
 * advantage of the processor cache */
void mem_pool_retake (node **pool_head, node *ptr)
{
   if (!ptr || !pool_head) return;
   ptr->next = *pool_head;
   *pool_head = ptr;
}

Use these calls instead of malloc/free. When memory gets thin... well, that's a discussion for tomorrow, maybe. :)

What we could do is see how we can optimize the memory pool creation. A simple idea could be used: Instead of writing the next pointer for all the elements in the beginning, we can keep a mark of 'how much we have written so far', in the pool definition. One example of such modification below:

typedef struct {
   node *initial_head;
   node *head;
   int maxelem;
   int maxused;
} mem_pool_t;

mem_pool_t *mem_pool_create (int maxelem)
{
   mem_pool_t *pool;
   int i;
   pool = (mem_pool_t *)malloc (sizeof (mem_pool_t));
   pool->head = (node *)malloc (sizeof (node)*maxelem);
   pool->initial_head = pool->head;
   pool->maxelem = maxelem;
   pool->maxused = 0;
   if (!pool->head) {
      free (pool);
      return NULL;
   }
   pool->head->next = pool->head + 1;
   return pool;
}

node *mem_pool_extract (mem_pool_t *pool)
{
   node *ptr;
   if (!pool_head || !*pool_head) return NULL;
   ptr = pool->head;
   pool->head = ptr->next;
   ptr->next = NULL;
   if (pool->head && pool->head - pool->initial_head >= pool->maxused) {
      pool->maxused ++;
      if (pool->maxelem>pool->maxused)
         pool->head->next = pool->head + 1;
      else
         pool->head->next = NULL;
   }
   return ptr;
}

void mem_pool_retake (mem_pool_t *pool, node *ptr)
{
   if (!ptr || !pool) return;
   ptr->next = pool->head;
   pool->head = ptr;
}

Update: I got a bit sloppier with the retake, updated the right code :)

A question of execution time

We all know that the complexity of algorithms we use influences a lot the performance of our programs.The difference between having linear time algorithms and having constant time algorithms is seen quite soon, after the first hundreds of items.

So here's a simple question. What's the complexity of adding an element at the end of a linked list? The naive answer usually is 'constant time'. But let's check the usual implementations.

Let's assume we have the structure:
typedef struct _node {
  int value;
  struct _node *next;
} node;

Implementation 1:
void add (node **head, int value)
{
   node *new_node, *ptr;
   if (!head) return;

   new_node = (node *)malloc (sizeof (node));
   new_node->next = NULL;
   new_node->value = value;
   ptr = *head;
   while (ptr && ptr->next) ptr = ptr->next;
   if (ptr)
      ptr->next = new_node;
   else
      *head = new_node;
}

This implementation is clearly NOT in constant time because, well, we have to start from the head, get all the way to the end of the list and add the element there. Time: O(n). At least. More later.

The obvious fix would be to also have the tail of the list. To make the tail available as well. So let's expand our data structures with a 'list' structure that will contain the head and the tail.

typedef struct {
   node *head;
   node *tail;
} list;

Implementation 2:
void add (list *l, int value)
{
   node *new_node, *ptr;
   if (!l) return;
   new_node = (node *)malloc (sizeof (node));
   new_node->next = NULL;
   new_node->value = value;
   if (l->tail) {
       l->tail->next = new_node;
       l->tail = new_node;
   } else {
       l->head = l->tail = new_node;
   }
}

Great! Problem solved! We have constant time for adding. Don't we? Begginer's luck, we're safe at bay!

...

...

Oh wait...

...

...

What is the execution time of malloc?

Oh sh**.

Good, now that we found out that it's not so easy to do that, we can say goodbye to the idea that a dynamically allocated linked list does not have the possibility to add an element in O(1) simply because we have to allocate the element. In academia it's a good and nice O(1) solution, in RealLife™ it's not. So what do we do?

The solutions I found so far are not perfect by far. First solution would be to have something like a smart pool that will give us the elements in a more or less constant time (one idea would be to have a buffer of more node structures, and then one could perform a circular search from the last 'allocated' node). Another solution would be to not do dynamic allocation at all, but that defeats the purpose of having a linked list in the first place.

One reminder. Never forget that just because you have a one liner doesn't mean that the execution time for that line is constant. And memory allocation can be one of the most critical operation for the performance of your program

Saturday, January 9, 2010

.bin/.cue to ISO or CDR (audio tracks)

Coming from the windows world, a lot of applications ;) come as .bin/.cue archives instead of the classical .iso format that we all love. Fortunately there is a simple solution called bchunk that will allow you to easily transform the bin/cue archive into an ISO archive. Invokation is simple:
bchunk file.bin file.cue file - and it will output file.format (ISO/UGH) or CDR audio tracks. :)
See the manual page for more infos. Nice tool, simple and effective. Loved it.

Saturday, January 2, 2010

(k)ubuntu StarCraft

I haven't played the game in quite a long while, and I said I should give it a shot, now that there's the new year and some time to waste.

The only way to do it cleanly on Linux is, of course, to use WINE, the other option being a virtual instance of Windows (which is not as fun as people might think. So I mounted the Starcraft CDs (Broodwar as well) installed them and played them.

First problem: no sound. After restarting with OSS checked as the default for wine in winecfg, the sound worked again - but unfortunately, I can't reproduce the reasons why the sound didn't work (this really sucks, does anyone listen?).

Second problem: after a short while, the mouse moved incredibly sluggish and it was a very unpleasant experience to play the game. This problem is a known problem among the WINE users, and after checking the winedb informations, I found out that you need to create the following registry entry:
HKEY_CURRENT_USER/Software/Wine/Direct3D , string entry: DirectDrawRenderer, value "opengl" (without the quotes). To edit this entry, run regedit. After running again the game, it worked smoothly.

Third problem: the disk kept spinning in my drive, and I really, really hate that. Actually, one thing I hate about the games I buy is that they force me to use their disk (with the notable exception of Guild Wars, which has an internet authentication, thus it doesn't need to do that). This was so annoying I had to search a no-cd patch for Starcraft. And I found it. Offered by Blizzard. It seems that if you apply the patch for version 1.15.2 or later (1.16.1 was the version I used) they let the game work without the CD - you just have to copy the file install.exe from the StarCraft original CD as the file "StarCraft.mpq" in the Starcraft installation folder, and the file install.exe from the BroodWar CD to the file "BroodWar.mpq". Again, no quotes.

One problem, and an annoying one, if you ask me. The patch will automatically close your session and throw you in the login screen. I don't understand why, so if possible (internet available) make sure that you patch your game via Battlenet rather than running the patch. :)

Run it, and it works. No CD, and smooth, just as you wanted it. En taro adun, Executor!

Now I wonder... Does Blizzard have a no-cd patch for Warcraft II as well? :)

PS: for Warcraft II you have to mount the disk by hand. My solution after I made the disk image was to have a script as such:
#!/bin/bash
sudo mount -o loop "$HOME/Games/Warcraft II BNE/war2bne.iso" "$HOME/Games/Warcraft II BNE/CD/"
pushd $HOME/Games/Warcraft\ II\ BNE
./Warcraft\ II\ BNE.exe