27.1.13

Hand Crafted G Code

The hair-brained idea I had to make a heat exchanger has yielded first artifacts.
The idea is to use two multi-section 3D printed half pipes for counter-flow air handling. The two half pipes would share a common bulkhead down the middle which would be perforated by diamond shaped holes. The holes would line up between the two half pipes, and be separated from each other by bits of tin foil glued to the edges to keep the air streams from mixing and provide good heat transfer. The two half pipes glued like this into one round pipe would be joined with others like it into a meter long (or more) exchanger, wrapped with insulation (fiberglass pink maybe) and the whole thing stuffed in a PVC water pipe for structural integrity.
In the original sketch, shown at the right, I was more worried about the method used for interlocking the sections and making sure there were no bumps along the bulkhead. Small sections are needed to accommodate the limited build area of the 3D printer. As it turns out the sections hold together well enough with just the texture of the plastic layers rubbing together, so the little raised bump on the lip and matching interior depression aren't really needed.
The big problem was, I discovered I couldn't use any of the existing slicing programs. There were two problems to overcome. One was getting the walls thin enough, and the other was keeping the G-code file size and generation time manageable. Even though plastic is a good insulator, you want thin walls so the heat isn't transmitted down the pipe and also so that it takes the least amount of plastic to create the pipes as possible. Some preliminary work I did when using the usual workflow of creating a 3D model, exporting it as an STL file, and then slicing the STL file to G-code was giving complete garbage multi-megabyte files and taking on the order of hours for simple structures.
So I decided to write my own program. For the wall thickness problem, you can craft it so that the wall thickness is exactly two extrudate traces wide. For the file size problem, using the G2 and G3 arc codes supported by the Marlin firmware makes for a very compact - and exact - representation of the geometry. Slicing programs are not able to use G2 and G3 because they start from an already tessellated STL file and hence would have to "recreate" the arc information from adjacent planar faces. What crap.
So I wrote the program in Python, to kill two birds with one stone - learn a little Python, and generate the g-code files for the pipes.
This is the first Python program I've written, so please forgive any idiotic constructs you see in the code below. It's not generic in any way. My excuse is that it's just enough to get my job done. The results are shown at left, with a couple of sections 50mm long and 40mm in diameter. The intent is to eventually make them four times bigger, otherwise the pressure losses from the air turbulence would be too great.
But the first problem is to figure out why one side of the hole is malformed as the printer climbs the 45° angle...


# coding=utf-8
'''
Pipe generator.
Created on January 27, 2013
@author: Derrick Oswald
'''
import math
import sys
import textwrap

class Machine:
    """A class representing parameters of a additive manufacturing 3D printer."""
    # g code preamble
    preamble = textwrap.dedent("""\
        M92 E865.888000    ; set axis_steps_per_unit to calibrated value
        M109 S210.000000   ; set extruder temperature and wait
        G21                ; metric values
        G90                ; absolute positioning
        M107               ; start with the fan off
        G28 X0 Y0          ; move X/Y to min endstops
        G28 Z0             ; move Z to min endstops
        G92 X0 Y0 Z0 E0    ; reset software position to front/left/z=0.0
        G1 Z15.0 F420
        G92 E0             ; zero the extruded length
        G1 F200 E3         ; extrude 3mm
        G92 E0             ; zero the extruded length again
        ; start""")
    # g code postamble
    postamble = textwrap.dedent("""\
        ; end
        M104 S0            ; extruder heater off
        M140 S0            ; heated bed heater off (if you have it)
        G91                ; relative positioning
        G1 Z2.5 E-5 F9000  ; heads up retracting
        G28 X0 Y0          ; move X/Y to min endstops, so the head is out of the way
        M84                ; steppers off
        G90                ; absolute positioning""")
    nozzle = 0.4 # nozzle diameter (mm)
    filament = 2.89 # filament diameter (mm)
    width = 0.5 # trace width or more likely the minimum wall thickness (mm)
    xorigin = 100.0 # X origin location (mm)
    yorigin = 100.0 # Y origin location (mm)
    zorigin = 0.0 # Z origin location (mm)
    eorigin = 0.0 # E origin location (mm)
    fast = 9000.0 # fast (slew) speed (mm/sec)
    slow = 3600.0 # slow (print) speed (mm/sec)
    def computeExtrudateCrossSecionalArea (self, layer = 0.1):
        # compute the area of the extrudate as a squashed circle cross section
        # with a thickness of layer and two rounded (bulgy) ends for a total of width
        return (layer * (self.width - layer) + (math.pi * (layer / 2.0) * (layer / 2.0)))

class Pipe:
    """A class representing the pipe to be created."""
    radius = 20.0 # inner radius of pipe (mm)
    wall = 2 # number of traces in the wall thickness
    length = 8.0 # pipe length (mm)
    overlap = 3.0 # pipe overlap joint size (mm)
    def computeR (self, machine, z = 0.0):
        thickness = self.wall * machine.width
        if z < self.overlap:
            r = self.radius + thickness
        elif z < self.overlap + thickness:
            r = self.radius + thickness - (z - self.overlap) # linear 45° interpolation
        else:
            r = self.radius
        return(r);
    def prepare (self, machine, job):
        area = machine.computeExtrudateCrossSecionalArea (job.layer)
        filament = machine.filament / 2.0 # radius
        x = machine.xorigin
        y = machine.yorigin
        z = machine.zorigin;
        e = machine.eorigin;
        print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' F' + str(machine.fast) + ' ; goto center stage'
        # draw a box around the working area by clearance amount
        halfedge = self.radius + job.clearance
        edge = halfedge * 2.0
        x += halfedge
        z += job.layer
        vol = (area * edge) / (math.pi * filament * filament) # E movement required to extrude a full edge
        print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' F' + str(machine.fast) + ' ; draw a box'
        y -= halfedge
        e += vol / 2.0
        print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
        x -= edge
        e += vol
        print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
        y += edge
        e += vol
        print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
        x += edge
        e += vol
        print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
        y -= halfedge
        e += vol / 2.0
        print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
        return (e)
    def generate(self, machine, job):
        if '' != machine.preamble:
            print >> job.output, machine.preamble
        e = self.prepare (machine, job)
        filament = machine.filament / 2.0 # radius
        area = machine.computeExtrudateCrossSecionalArea (job.layer)
        x = machine.xorigin
        y = machine.yorigin
        z = machine.zorigin;
        while z < self.length:
            z += job.layer
            radius = self.computeR(machine, z)
            for i in range (self.wall):
                r = radius + machine.width * i + machine.width / 2.0
                x = machine.xorigin + r
                y = machine.yorigin + 0.0
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' F' + str(machine.fast)
                trace = math.pi * r
                vol = (area * trace) / (math.pi * filament * filament) # E movement required to extrude a half circle
                e += vol
                x -= 2.0 * r
                print >> job.output, 'G3 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' I' + str(-r) + ' J' + str(0.0) + ' F' + str(machine.slow)
                e += vol
                x += 2.0 * r
                print >> job.output, 'G3 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' I' + str(+r) + ' J' + str(0.0) + ' F' + str(machine.slow)
        if '' != machine.postamble:
            print >> job.output, machine.postamble

class HalfPipe (Pipe):
    """A class representing the half pipe to be created."""
    gapedge = 2.0 # gap limiting edge width (mm)
    def computeD(self, machine, z, i):
        thickness = self.wall * machine.width
        d = thickness - (machine.width / 2.0) - (i * machine.width)
        startz = self.length - (self.overlap + thickness)
        if z > startz:
            delta = z - startz # linear 45° interpolation
            if (delta > thickness):
                delta = thickness
            d += delta
        return(d)
    def computeGap(self, machine, z): # actually the half-gap
        thickness = self.wall * machine.width
        startz = self.overlap + thickness
        endz = self.length - (self.overlap + thickness)
        maximum = self.computeR(machine, z) + machine.width / 2.0 - self.gapedge
        if ((z > startz) and (z < endz)):
            glo = z - startz
            ghi = endz - z
            if (ghi < glo):
                g = ghi
            else:
                g = glo
            if (g > maximum):
                g = maximum
        else:
            g = 0.0
        return (g)
    def generate(self, machine, job):
        if '' != machine.preamble:
            print >> job.output, machine.preamble
        e = self.prepare (machine, job)
        filament = machine.filament / 2.0 # radius
        area = machine.computeExtrudateCrossSecionalArea (job.layer)
        x = machine.xorigin
        y = machine.yorigin
        z = machine.zorigin;
        while z < self.length:
            z += job.layer
            radius = self.computeR(machine, z)
            gap = self.computeGap(machine, z)
            if ((0 != gap) and (2 == self.wall)):
                r0 = radius +                 machine.width / 2.0
                r1 = radius + machine.width + machine.width / 2.0
                d0 = self.computeD(machine, z, 0)
                d1 = self.computeD(machine, z, 1)
                dx0 = math.sqrt(r0 * r0 - d0 * d0)
                dx1 = math.sqrt(r1 * r1 - d1 * d1)
                x = machine.xorigin + dx0
                y = machine.yorigin + d0
                vol0 = (area * math.pi * (r0 - 2 * d0)) / (math.pi * filament * filament) # E movement required to extrude an inner half circle, approximately
                vol1 = (area * math.pi * (r1 - 2 * d1)) / (math.pi * filament * filament) # E movement required to extrude an outer half circle, approximately
                vol2 = (area * 2 * dx0) / (math.pi * filament * filament) # E movement required to extrude an inner diameter
                vol3 = (area * 2 * dx1) / (math.pi * filament * filament) # E movement required to extrude an outer diameter
                vol4 = (area * machine.width) / (math.pi * filament * filament) # E movement required to extrude across from inner to outer wall
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' F' + str(machine.fast)
                e += vol0
                x -= 2 * dx0
                print >> job.output, 'G3 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' I' + str(-dx0) + ' J' + str(-d0) + ' F' + str(machine.slow)
                e += vol2 * (dx0 - gap) / dx0
                x += dx0 - gap
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
                e += vol4
                y -= machine.width
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
                e += vol3 * (dx1 - gap) / dx1
                x -= dx1 - gap
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
                e += vol1
                x += 2 * dx1
                print >> job.output, 'G2 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' I' + str(dx1) + ' J' + str(-d1) + ' F' + str(machine.slow)
                e += vol3 * (dx1 - gap) / dx1
                x -= dx1 - gap
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
                e += vol4
                y += machine.width
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
                e += vol2 * (dx0 - gap) / dx0
                x += dx0 - gap
                print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
            else:
                for i in range (self.wall):
                    r = radius + machine.width * i + machine.width / 2.0
                    d = self.computeD(machine, z, i)
                    dx = math.sqrt(r * r - d * d)
                    x = machine.xorigin + dx
                    y = machine.yorigin + d
                    vol = (area * math.pi * (r - 2 * d)) / (math.pi * filament * filament) # E movement required to extrude a half circle, approximately
                    vol2 = (area * 2 * dx) / (math.pi * filament * filament) # E movement required to extrude a diameter
                    print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' F' + str(machine.fast)
                    e += vol
                    x -= 2 * dx
                    print >> job.output, 'G3 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' I' + str(-dx) + ' J' + str(-d) + ' F' + str(machine.slow)
                    e += vol2
                    x += 2.0 * dx
                    print >> job.output, 'G1 X' + str(x) + ' Y' + str(y) + ' Z' + str(z) + ' E' + str(e) + ' F' + str(machine.slow)
        if '' != machine.postamble:
            print >> job.output, machine.postamble
    
class Job:
    """Parameters related to a specific print job."""
    output = sys.stdout # target for print
    layer = 0.1 # layer height (mm)
    clearance = 5.0 # offset outside of object for box (mm)

if __name__ == '__main__':
    machine = Machine ()
    pipe = HalfPipe ()
    pipe.radius = 20.0
    pipe.length = 50.0
    pipe.wall = 2
    pipe.overlap = 4.0
    job = Job()
    job.output = file ('output.gc', 'w')
    job.layer = 0.1
    job.clearance = 5.0
    pipe.generate (machine, job)
    job.output.flush()
    job.output.close()

20.1.13

Thing Tracker

The Thing Tracker Network got me thinking about exactly what would need to be shared. This immediately brought to mind the classic science fiction work "The Big Front Yard" by Clifford Simak, describing how interstellar trade must be in ideas and information and not material objects. The protagonist trades the "idea of paint" for the "idea of anti-gravity saddles".

What ideas do we have to trade?

I fall back to my design roots, and this forces me to ask what are the use-cases we wish to address? Let me start out with a few off the top of my head.

  • Replacement Part

    Aside from trivial handles and feet, printing replacement parts will probably need the participation of manufacturers. Some, like Nokia, have started this process already. But if you've ever been associated with product lifecycle management, you'll know there is a lot more to it than a one-time publication. Specifically, different product models, sizes, component part differences, configurations, geographies (line voltages, certification, and so on), versions over time, etc. make the correct part difficult to determine even for the manufacturers building it. All of this information would need to be included with the tracked object.

  • Custom Object

    Although there are ways to tailor objects after construction, there will be a use-case for things that are specifically built to fit people, pets, rooms, vehicles, etc. The big problem here is measuring and applying the measurements correctly to the objects. For this purpose, the instructions and parameters would need to be included with the tracked object so it's useful to the receiver.

  • Toys, Jewelry, Art

    By and large, this is the current set of objects, that requires only geometry, material, and some images and assembly instructions.

The current state of affairs is pretty crude. The existing sites like Thingiverse, Physible Exchange, The Pirate Bay, Archive of Our Own, etc. share only .stl files, or whatever else the author chooses to share. These are doomed eventually, because they

  1. don't have full parametric model files
  2. have insufficient metadata
  3. aren't indexed, other than by full text search or user specified ad-hoc tags
  4. aren't versioned
  5. are centralized to one site

The Thing Tracker may answer some of these. For example, it may be decentralized ... as long as the implementation uses something like torrent magnet files. I think the content and use-cases need to be determined and then the implementation will derive naturally from those. Are there any use-cases I omitted?

Carbon Offsetting

I don't trust CO2 carbon offset schemes. Just like every other charity, there is an efficiency rating for the process, and in my opinion the amount of good being done is minimal, if not negative.

Rather than try to ameliorate the situation with a band-aid solution, just use less fossil fuels in the first place. If you can't avoid flying, take it easy on the environment by bringing less with you.

For those who have to sit on their suitcases to close them, there is no salvation. But for normal people consider that it takes 2.5 liters of jet fuel to ship an extra kilogram across the Atlantic. A CO2 conversion chart says 2.52kg of CO2 is produced for every liter of jet fuel. This means you can do 6.3kg of CO2 offsetting by just not carrying an extra kilo.

To get that fuel consumption number took a little digging, and I present it here so others can ratify it if they want.

From Figure 4.8 Fuel Efficiency, the industry achieved 1630 Revenue Ton-Miles / 1000 liters in 2006. Revenue Ton-Miles is a measure of cargo amount. It is calculated by multiplying the weight in tons of the shipment being transported by the number of miles that it is transported. This metric includes only paid tonnage. In other words, only materials being transported for and being paid for by customers. Using 1.60934 kilometers/mile and 907.185 kg/ton, this works out to 4.2e-4 liters / kilogram-kilometer. For 6000 km trans Atlantic flight, you get about 2.5 liters / kilogram.

To put this in perspective, one penny is 2.35g, which needs 0.006 liters to cross the Atlantic. Using the current IATA Jet Fuel price of $3.08/(US)gallon (€0.61/liter), it costs nearly half a cent (0.48¢) to ship a penny. The airlines would be more environmentally friendly by converting your pocket change into paper bills just before you step on the aircraft. They could do this at a very customer advantageous exchange rate and still make money.

The airlines are on the right track in reducing luggage allowance and charging €75 for the extra 23kg piece of luggage ($100/50lbs). The next step is to charge per kilo of person. Fortunately for me, they don't do that yet.

The plane manufacturers can obviously do a lot more too, since 50-80% of the takeoff weight of an aircraft is just the plane.

13.1.13

Friction Welding

Following on post and a YouTube video by Fran, I tried my hand at friction welding. It works, but you need to practice.

Chuck up a straightened piece of ABS filament about five centimeter long in a Dremel tool such that about three or four centimeters are visible. Turn on the dremel tool at medium speed and drag the spinning plastic backwards along the joint to be welded making little circular motions.

The plastic will liquefy, there's a bit of a burnt plastic smell and then the parts are joined. The laminations of the printed part may come apart if you're too rough with it, so practice on some scraps first.

Follow welding best practices where possible, i.e. it's best in a corner between two pieces to be joined, and avoid butt welding like that shown below.

Backlash

Yesterday I discovered a significant amount of backlash in my Ultimaker's axes. This is how I fixed it.

First I isolated where the backlash was coming from. I used a bit of blue tape and stuck it to shafts and belts in the mechanical linkage while driving the axis back and forth with the smallest step in PrintRun. The reason for the blue tape is to amplify the small angular or linear motions enough to see them.

Starting at the motor shaft, observe the motion of the blue tape while clicking the small step left and right (X) or up and down (Y) in PrintRun. If you observe motion in both directions move to the next part in the linkage (pulley, belt, shaft). At some point in the process, the blue tape doesn't move unless you click the step button several times. The joint between the current part and the previous part is the problem.

For me, this was right at the beginning, either where the pulleys are set-screwed onto the motor shafts or the small pulley tension is set by the four motor mounting screws. So I took off the motor, removed and reattached the pulley on the motor shaft (to let the set screw get a new grip on the shaft), and reassembled it with a lot more tension in the small belts. I was able to remove most of the backlash just from that.

This also explains the flat spots I was getting when using G2 and G3 arc codes.

12.1.13

Microscopic

My eyes aren't so good anymore, so I have a head mounted magnifier that I got from Lee Valley some years ago. But for the current work I'm doing, I needed to get even more magnification. It turned out to be surprisingly easy.

I remember reading that by reversing the lens in a webcam, you can make a USB microscope. I had an old Logitech webcam lying around since my new laptop has an integrated one, and there's instructions on Instructables.

It took about 10 minutes before I was seeing stuff in the Quick View panel with the lens bodged on backwards with blue tape.

The trouble is, the position of the camera and specimen need to be micro-adjusted and then held in one position for long periods of time... hey wait a minute I have a X,Y,Z stage with just those properties.

So clamping the "microscope" onto the print head and running PrintRun to position it turns out to be a very good control system to look at your samples (if you turn the printhead fan off). It also shows some other interesting info.

For example, I now realize how much backlash I've got in my axes, because the image from the microscope is exquisitely sensitive to even the smallest motion. So how many times do you need to touch the 0.1mm step button in one direction to see motion after taking a 1mm step in the other direction. This I'll need to experiment with, so more on that later.

But I did find out what I wanted to find out - the two traces I was laying down were not touching. I'm guessing at the numbers here, but I think the magnification is about 300 times in the picture below, and the PLA trace running diagonally up from left to right (it looks like icing that's been scraped with a spatula) is about 0.35mm wide and is separated from the next trace in the lower right hand corner by 0.13mm.