I ran into another interesting memory issue while working on the Photobooth app. This time I was testing a template that took 4 photos, and right as the photos finished and I pushed the images to the print view controller, the app crashed. The console message said something about memory issues, so I immediately booted up trusty ol’ Instruments, and reran.

Turns out, when the print view controller is loading, my app was trying to allocate a whopping 1.26 to 1.5 GB of memory. What!!

screen-shot-2017-07-19-at-11-37-29-pm

Digging in into the Call Tree, the culprit was UIGraphicsBeginImageContextWithOptions, and its submethod CGBitmapContextCreate.

Looking through my code and commenting things out and rerunning, this was the innocuous line that caused all the trouble:

screen-shot-2017-07-19-at-11-52-41-pm

UIGraphicsBeginImageContextWithOptions(size, NO, 0.0);

The key part is the scale. Setting a scale of 0 is to use the default phone scale. In my case, I was testing with an iPhone 6+, so that means 3x. The function calling this creates the final image that gets printed, and I was using the same resolution as the original camera image, so about 4860 x 7025. At 3x, that’s:

4860px * 7025px * 3 * 1 byte/pixel = 102,424,500 bytes ~ 100 mb

which is apparently blown by up by 10x to create the image context. I don’t really understand what’s happening underneath here.

Since this wasn’t an image I was displaying at full resolution on the phone though, 3x scale is unnecessary so I changed it to 1x, and memory usage became a lot more reasonable, about 100 mb.… Read the rest

I’ve been doing some lower level graphics stuff recently for my Photo Booth app and ran into some interesting memory leaks. I had a feeling something was off so ran Instruments after coding, and sure enough, leaks were detected. Several actually. The culprits:

CGPathRef

The biggest offender by far. Forgetting to call CGPathRelease(path) after any form of CGPathCreate…() was the most comment mistake. An interesting case also came up with holding onto an instance variable of the path. So I had this path that was expensive to calculate and gets reused, so I save it into an instance variable:

@property (nonatomic) CGPathRef scaledActivePath;

But when it does get recalculated, I want to release the previous saved path, so I override the setter:

- (void)setScaledActivePath:(CGPathRef)scaledActivePath {
    CGPathRelease(_scaledActivePath);
    
    _scaledActivePath = scaledActivePath;
}

And finally, it needs to be released when the view controller is finished with:

- (void)dealloc {
    CGPathRelease(self.scaledActivePath);
}

 

Image construction with malloc()

In this case, I’m creating a color picker by calculating r, g, b values and placing them into a UInt8 * array, which is then used to create a CGDataProviderRef and CGImageRef. 

The final result looked great:

screen-shot-2017-07-11-at-11-31-37-pm

The code involved looked like:

NSInteger width = self.shadePicker.frame.size.width;
NSInteger height = self.shadePicker.frame.size.height;
NSInteger dataLength = width * height * 4;
self.shadeData = (UInt8 *) malloc(dataLength * sizeof(UInt8));

for (int column = 0; column < width; column++) {
    for (int row = 0; row < height; row++) {
        long i = (column + row * width) * 4;
        
        CGFloat r;
        CGFloat g;
        CGFloat b;
        
        [self colorForBaseHue:hue
                       column:column row:row
                        width:width height:height
                            r:&r g:&g b:&b];
        
        self.shadeData[i] = r;
        self.shadeData[i+1] = g;
        self.shadeData[i+2] = b;
        self.shadeData[i+3] = 255.0;
    }
}

CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
CGDataProviderRef dataProvider = CGDataProviderCreateWithData(NULL, self.shadeData, dataLength, NULL);
CGImageRef imgRef = CGImageCreate(width, height, 8, 32, 4 * width, colorSpace,kCGImageByteOrder32Big | kCGImageAlphaPremultipliedLast, dataProvider, NULL, true, kCGRenderingIntentDefault);
UIImage *img = [UIImage imageWithCGImage:imgRef];

CGImageRelease(imgRef);
CGColorSpaceRelease(colorSpace);
CGDataProviderRelease(dataProvider);

Notice what’s missing?… Read the rest

Another birthday in my 30’s, and I’m sick this time, which isn’t very auspicious. All the planning and excitement about going to Tahoe to snowboard on my birthday vanished when I began feeling weak the day before, then completely useless the day of.

The human body is so weak when sick. I barely had enough energy to stand and brush my teeth. I had to lean against the counter, closing my eyes as I brushed because it was 9pm and I was already exhausted. How much it reminded me that I’ve wasted so much time and how little time I have left.

If I got lucky, and lived until about 65, the age at which the body is still able to perform when not sick, then I’ve got about 35 good years left.

Minus 3 years for sickness, and I’ve got 32 years.

Minus 12 years of birthing and childcare, assuming 2 children, and I’ve got 20 left.

God, I’ve wasted so much time.  Only 20 years left to do everything I still dream of doing. 20 years to travel the world, painting the scenes that I see. Code all the programs I think of. Start a company, fail, keep trying at it, maybe succeed, maybe not, but at least I tried. Climb mountains, meet people from different walks all over the world, try different lifestyles. I want to make a video game, doesn’t have to be flashy, but something fun and original. Buy some land, build my own house. Try different sports. Go back to the same beautiful spot in the woods everyday to paint a picture.… Read the rest