This week my partner had design a first sample for mobile robot so had to create a programming for the circuit to show a basic movement of the robot such move left, right, forward, reverse and stop.
Figure below show the circuit design:
And this is the programming for the circuit:
#include <16f84a.h>
#fuses HS,NOWDT,PUT
#use delay(clock=20000000)
void main()
{
while(TRUE)
{
if(!input(pin_a0))
{
output_b(0x68);
delay_ms(3000);
output_b(0x6A);
}
else if(!input(pin_a1))
{
output_b(0x62);
delay_ms(3000);
output_b(0x6A);
}
else if(!input(pin_a2))
{
output_b(0x6A);
}
else if(!input(pin_a3))
{
output_b(0x74);
}
else if(!input(pin_a4))
{
output_b(0x00);
}
}
}
In this programming, i had use if-else function for the selection of input switch and the output controlled by sending a data to the whole port.
Final Year Project Logbook
Thursday, October 14, 2010
Saturday, October 9, 2010
Week 11 Multilayer Perceptron
This week i learn about multilayer perceptron, where a single layer perceptron only have an input layer and output layer, this multilayer layer perceptron have another layer inbetween input and output layer which is hidden layer.
Figure 1 : Multilayer perceptron
The Multilayer Perceptron
The multilayer perceptron (MLP) is a hierarchical structure of several perceptrons, and overcomes the shortcomings of these single-layer networks.
The multilayer perceptron is an artificial neural network that learns nonlinear function mappings. The multilayer perceptron is capable of learning a rich variety of nonlineardecision surfaces.
Nonlinear functions can be represented by multilayer perceptrons with units that use nonlinear activation functions. Multiple layers of cascaded linear units still produce only linear mappings.
Differentiable Activation Functions
The training algorithm for multilayer networks requires differentiable, continuous nonlinear activation functions. Such a function is the sigmoid, or logistic function:
o = s ( s ) = 1 / ( 1 + e-s )
where s is the sum: s=S i=0d wi xi of products from the weights wi and the inputs xi.
Sometimes s is called alternatively squashing function as it maps a very large input domain to a small range of outputs.
Another nonlinear function often used in practice is the hyperbolic tangent:
Sometimes the hyperbolic tangent is preferred as it makes the training a little easier.
Figure 1 : Multilayer perceptron
The Multilayer Perceptron
The multilayer perceptron (MLP) is a hierarchical structure of several perceptrons, and overcomes the shortcomings of these single-layer networks.
The multilayer perceptron is an artificial neural network that learns nonlinear function mappings. The multilayer perceptron is capable of learning a rich variety of nonlineardecision surfaces.
Nonlinear functions can be represented by multilayer perceptrons with units that use nonlinear activation functions. Multiple layers of cascaded linear units still produce only linear mappings.
Differentiable Activation Functions
The training algorithm for multilayer networks requires differentiable, continuous nonlinear activation functions. Such a function is the sigmoid, or logistic function:
where s is the sum: s=S i=0d wi xi of products from the weights wi and the inputs xi.
Sometimes s is called alternatively squashing function as it maps a very large input domain to a small range of outputs.
Another nonlinear function often used in practice is the hyperbolic tangent:
o = tanh( s ) = ( es - e-s ) / (es + e-s)
Tuesday, October 5, 2010
Week 10 Neural Network
This week i had learn about Neural Network where it is another method to apply artificial intelligent in a mobile robot programming.
Introduction
An artificial neural network (ANN), usually called "neural network" (NN), is a mathematical model or computational model that is inspired by the structure and/or functional aspects of biological neural networks. It consists of an interconnected group of artificial neurons and processes information using a connectionist approach to computation. In most cases an ANN is an adaptive system that changes its structure based on external or internal information that flows through the network during the learning phase. Modern neural networks are non-linear statistical data modeling tools. They are usually used to model complex relationships between inputs and outputs or to find patterns in data.
Introduction
An artificial neural network (ANN), usually called "neural network" (NN), is a mathematical model or computational model that is inspired by the structure and/or functional aspects of biological neural networks. It consists of an interconnected group of artificial neurons and processes information using a connectionist approach to computation. In most cases an ANN is an adaptive system that changes its structure based on external or internal information that flows through the network during the learning phase. Modern neural networks are non-linear statistical data modeling tools. They are usually used to model complex relationships between inputs and outputs or to find patterns in data.
How the Human Brain Learns?
Much is still unknown about how the brain trains itself to process information, so theories abound. In the human brain, a typical neuron collects signals from others through a host of fine structures called dendrites. The neuron sends out spikes of electrical activity through a long, thin stand known as an axon, which splits into thousands of branches. At the end of each branch, a structure called a synapse converts the activity from the axon into electrical effects that inhibit or excite activity from the axon into electrical effects that inhibit or excite activity in the connected neurones. When a neuron receives excitatory input that is sufficiently large compared with its inhibitory input, it sends a spike of electrical activity down its axon. Learning occurs by changing the effectiveness of the synapses so that the influence of one neuron on another changes.A simple neuron
An artificial neuron is a device with many inputs and one output. The neuron has two modes of operation; the training mode and the using mode. In the training mode, the neuron can be trained to fire (or not), for particular input patterns. In the using mode, when a taught input pattern is detected at the input, its associated output becomes the current output. If the input pattern does not belong in the taught list of input patterns, the firing rule is used to determine whether to fire or not.A simple neuron
A more complicated neuron
The previous neuron doesn't do anything that conventional conventional computers don't do already. A more sophisticated neuron (figure 2) is the McCulloch and Pitts model (MCP). The difference from the previous model is that the inputs are 'weighted', the effect that each input has at decision making is dependent on the weight of the particular input. The weight of an input is a number which when multiplied with the input gives the weighted input. These weighted inputs are then added together and if they exceed a pre-set threshold value, the neuron fires. In any other case the neuron does not fire.Figure 2. An MCP neuron
In mathematical terms, the neuron fires if and only if; X1W1 + X2W2 + X3W3 + ... > T
The addition of input weights and of the threshold makes this neuron a very flexible and powerful one. The MCP neuron has the ability to adapt to a particular situation by changing its weights and/or threshold. Various algorithms exist that cause the neuron to 'adapt'; the most used ones are the Delta rule and the back error propagation. The former is used in feed-forward networks and the latter in feedback networks.
Friday, September 24, 2010
Week 9 Revision of C programming II
This week also i had do a revision on c programming buta bit more focused on the programming language,
Introduction
In C, programs are composed of statements. These statements are terminated with a semi-colon, and are collected in sections known as functions. By convention, a statement should be kept on its own line, as shown in the example below:
The following block of code is essentially the same: while it contains exactly the same code, and will compile and execute with the same result, the removal of spacing causes an essential difference making it harder to read:
The simple use of indents and line breaks can greatly improve the readability of the code; without making any impact whatsoever on how the code performs. By having readable code, it is much easier to see where functions and procedures end, and which lines are part of which loops and procedures.
This book is going to focus on the above piece of code, and how to improve it. Please note that during the course of the tutorial, there will be many (apparently) redundant pieces of code added. These are only added to provide examples of techniques that we will be explaining, without breaking the overall flow of code that the program achieves.
The addition of white space inside your code is arguably the most important part of good code structure. Effective use of white space can create a visual scale of how your code flows, which can be very important when returning to your code when you want to maintain it.
With minimal line breaks, code is barely readable by humans, and may be hard to debug or understand:
Rather than putting everything on one line, it is much more readable to break up long lines so that each statement and declaration goes on its own line. After inserting line breaks, the code will look like this:
The following lines of code have line breaks between functions, but without indentation.
But this still isn't as readable as it can be.
It is now fairly obvious as to which parts of the program fit inside which blocks. You can tell which parts of the program the coder has intended to loop, and which ones he has not. Although it might not be immediately noticeable, once many nested loops and paths get added to the structure of the program, the use of indentation can be very important. This indentation makes the structure of your program clear.
Indentation was originally one tab character, or the equivalent of 8 spaces. Research since the original indent size has shown that indents between 2 to 4 characters are easier to read, resulting in such tab sizes being used as default in modern IDEs. However, an indent of 8 characters may still be in use for some systems.
Comments in modern flavors of C (and many other languages) can come in two forms:
and
Note that Single line comments are a fairly recent addition to C, so some compilers may not support them. A recent version of GCC will have no problems supporting them.
This section is going to focus on the various uses of each form of commentary.
Based on our previous program, there are two good places to place comments
As an example, suppose we had a program that was designed to print "Hello, World! " a certain number of times, a specified number of times. There would be many for loops in this program. For this example, we shall call the number of lines i, and the number of strings per line as j.
A good example of a multi-line comment that describes 'for' loop i's purpose would be:
This provides a good explanation of what i's purpose is, whilst not going into detail of what j does. By going into detail over what the specific path does (and not ones inside it), it will be easier to troubleshoot the path.
Similarly, you should always include a multi-line comment before each function, to explain the role, preconditions and postconditions of each function. Always leave the technical details to the individual blocks inside your program - this makes it easier to troubleshoot.
A function descriptor should look something like:
This system allows for an at-a-glance explanation of what the function should do. You can then go into detail over how each aspect of the program is achieved later on in the program.
Finally, if you like to have aesthetically-pleasing source code, the multi-line comment system allows for the easy addition comment boxes. These make the comments stand out much more than they would without otherwise. They look like this.
Applied to our original program, we can now include a much more descriptive and readable source code:
This will allow any outside users of the program an easy way to understand what the code does, and how it works. It also prevents confusion with other like-named functions.
A few programmers add a column of stars on the right side of a block comment:
But most programmers don't put any stars on the right side of a block comment. They feel that aligning the right side is a waste of time.
Comments written in source files can be used for documenting source code automatically by using popular tools like Doxygen
Introduction
In C, programs are composed of statements. These statements are terminated with a semi-colon, and are collected in sections known as functions. By convention, a statement should be kept on its own line, as shown in the example below:
#include <stdio.h> int main(void) { printf("Hello, World!\n"); return 0; }
#include <stdio.h> int main(void) {printf("Hello, World!\n");return 0;}
This book is going to focus on the above piece of code, and how to improve it. Please note that during the course of the tutorial, there will be many (apparently) redundant pieces of code added. These are only added to provide examples of techniques that we will be explaining, without breaking the overall flow of code that the program achieves.
Line Breaks and Indentation
The addition of white space inside your code is arguably the most important part of good code structure. Effective use of white space can create a visual scale of how your code flows, which can be very important when returning to your code when you want to maintain it.
Line Breaks
Note that we have used line numbers here; they are not a part of the actual code. They are only there for reference in this book. |
-
#include <stdio.h>
-
int main(void){ int i=0; printf("Hello, World!"); for (i=0; i<1; i++){ printf("\n"); break; } return 0; }
#include <stdio.h> int main(void) { int i=0; printf("Hello, World!"); for (i=0; i<1; i++) { printf("\n"); break; } return 0; }
Blank Lines
Blank lines should be used to offset the main components of your code. Use them- After precompiler declarations.
- After new variables are declared.
- Between lines 1 and 2, because line 1 has a preprocessor directive
- Between lines 4 and 5, because line 4 contains a variable declaration
The following lines of code have line breaks between functions, but without indentation.
-
#include <stdio.h>
-
int main(void)
-
{
-
int i=0;
-
printf("Hello, World!");
-
for (i=0; i<1; i++)
-
{
-
printf("\n");
-
break;
-
}
-
return 0;
-
}
Indentation
Many text editors automatically indent appropriately when you hit the enter/return key.
Although adding simple line breaks between key blocks of code can make code easier to read, it provides no information about the block structure of the program. Using the tab key can be very helpful now: indentation visually separates paths of execution by moving their starting points to a new column in the line. This simple practice will make it much easier to read and understand code. Indentation follows a fairly simple rule:- All code inside a new block should be indented by one tab more than the code in the previous path.
- Lines 5 to 13
- Lines 10 and 11
-
#include <stdio.h>
-
int main(void)
-
{
-
int i=0;
-
printf("Hello, World!");
-
for (i=0; i<1; i++)
-
{
-
printf("\n");
-
break;
-
}
-
return 0;
-
}
Indentation was originally one tab character, or the equivalent of 8 spaces. Research since the original indent size has shown that indents between 2 to 4 characters are easier to read, resulting in such tab sizes being used as default in modern IDEs. However, an indent of 8 characters may still be in use for some systems.
Comments
Comments in code can be useful for a variety of purposes. They provide the easiest way to set off specific parts of code (and their purpose); as well as providing a visual "split" between various parts of your code. Having a good comments throughout your code will make it much easier to remember what specific parts of your code do.Comments in modern flavors of C (and many other languages) can come in two forms:
//Single Line Comments (added by C99 standard,famously known as c++ style of comments)
/*Multi-Line Comments*/ (only form of comments supported by C89 standard)
This section is going to focus on the various uses of each form of commentary.
Single-line Comments
Single-line comments are most useful for simple 'side' notes that explain what certain parts of the code do. The best places to put these comments are next to variable declarations, and next to pieces of code that may need explanation.Based on our previous program, there are two good places to place comments
- Line 3, to explain what 'int i' is going to do
- Line 11, to explain why there is a 'break' keyword.
#include <stdio.h> int main(void) { int i=0; // loop variable. printf("Hello, World!"); for (i=0; i<1; i++) { printf("\n"); break; //Exits 'for' loop. } return 0; }
Multi-line Comments
Single-line comments are a new feature, so many C programmers only use multi-line comments.
Multi-line comments are most useful for long explanations of code. They can be used as copyright/licensing notices, and they can also be used to explain the purpose of a block of code. This can be useful for two reasons: They make your functions easier to understand, and they make it easier to spot errors in code. If you know what a block is supposed to do, then it is much easier to find the piece of code that is responsible if an error occurs.As an example, suppose we had a program that was designed to print "Hello, World! " a certain number of times, a specified number of times. There would be many for loops in this program. For this example, we shall call the number of lines i, and the number of strings per line as j.
A good example of a multi-line comment that describes 'for' loop i's purpose would be:
/* For Loop (int i)
Loops the following procedure i times (for number of lines). Performs 'for' loop j on each loop,
and prints a new line at end of each loop.
*/
Similarly, you should always include a multi-line comment before each function, to explain the role, preconditions and postconditions of each function. Always leave the technical details to the individual blocks inside your program - this makes it easier to troubleshoot.
A function descriptor should look something like:
/* Function : int hworld (int i,int j)
Input : int i (Number of lines), int j (Number of instances per line)
Output : 0 (on success)
Procedure: Prints "Hello, World!" j times, and a new line to standard output over i lines.
*/
Finally, if you like to have aesthetically-pleasing source code, the multi-line comment system allows for the easy addition comment boxes. These make the comments stand out much more than they would without otherwise. They look like this.
/***************************************
* This is a multi line comment
* That is nearly surrounded by a
* Cool, starry border!
***************************************/
#include <stdio.h> int main(void) { /************************************************************************************ * Function: int main(void) * Input : none * Output : Returns 0 on success * Procedure: Prints "Hello, World!" and a new line to standard output then exits. ************************************************************************************/ int i=0; //Temporary variable used for 'for' loop. printf("Hello, World!"); /* FOR LOOP (int i) Prints a new line to standard output, and exits */ for (i=0; i<1; i++) { printf("\n"); break; //Exits 'for' loop. } return 0; }
A few programmers add a column of stars on the right side of a block comment:
/***************************************
* This is a multi line comment *
* That is completely surrounded by a *
* Cool, starry border! *
***************************************/
Comments written in source files can be used for documenting source code automatically by using popular tools like Doxygen
Thursday, September 16, 2010
Week 8 Revision of C programming
This week i will do a revision on c programming. This is because i need to create a program as an example for the mobile robot, this program will make the mobile robot do a basic movement.
The stages of developing your C program are as follows.
Creating the program
Create a file containing the complete program, such as the above example. You can use any ordinary editor with which you are familiar to create the file. One such editor is textedit available on most UNIX systems.
The filename must by convention end ``.c'' (full stop, lower case c), e.g. myprog.c or progtest.c. The contents must obey C syntax. For example, they might be as in the above example, starting with the line /* Sample .... (or a blank line preceding it) and ending with the line } /* end of program */ (or a blank line following it).
Compilation
There are many C compilers around. The cc being the default Sun compiler. The GNU C compiler gcc is popular and available for many platforms. PC users may also be familiar with the Borland bcc compiler.
There are also equivalent C++ compilers which are usually denoted by CC (note upper case CC. For example Sun provides CC and GNU GCC. The GNU compiler is also denoted by g++
Other (less common) C/C++ compilers exist. All the above compilers operate in essentially the same manner and share many common command line options. Below and in Appendix [*] we list and give example uses many of the common compiler options. However, the best source of each compiler is through the online manual pages of your system: e.g. man cc.
For the sake of compactness in the basic discussions of compiler operation we will simply refer to the cc compiler -- other compilers can simply be substituted in place of cc unless otherwise stated.
To Compile your program simply invoke the command cc. The command must be followed by the name of the (C) program you wish to compile. A number of compiler options can be specified also. We will not concern ourselves with many of these options yet, some useful and often essential options are introduced below.
Thus, the basic compilation command is:
cc program.c
where program.c is the name of the file.
If there are obvious errors in your program (such as mistypings, misspelling one of the key words or omitting a semi-colon), the compiler will detect and report them.
There may, of course, still be logical errors that the compiler cannot detect. You may be telling the computer to do the wrong operations.
When the compiler has successfully digested your program, the compiled version, or executable, is left in a file called a.out or if the compiler option -o is used : the file listed after the -o.
It is more convenient to use a -o and filename in the compilation as in
cc -o program program.c
which puts the compiled program into the file program (or any file you name following the "-o" argument) instead of putting it in the file a.out .
Running the program
The next stage is to actually run your executable program. To run an executable in UNIX, you simply type the name of the file containing it, in this case program (or a.out)
This executes your program, printing any results to the screen. At this stage there may be run-time errors, such as division by zero, or it may become evident that the program has produced incorrect output.
If so, you must return to edit your program source, and recompile it, and run it again.
The C Compilation Model
We will briefly highlight key features of the C Compilation model (Fig. 2.1) here.
The Preprocessor accepts source code as input and is responsible for
The stages of developing your C program are as follows.
Creating the program
Create a file containing the complete program, such as the above example. You can use any ordinary editor with which you are familiar to create the file. One such editor is textedit available on most UNIX systems.
The filename must by convention end ``.c'' (full stop, lower case c), e.g. myprog.c or progtest.c. The contents must obey C syntax. For example, they might be as in the above example, starting with the line /* Sample .... (or a blank line preceding it) and ending with the line } /* end of program */ (or a blank line following it).
Compilation
There are many C compilers around. The cc being the default Sun compiler. The GNU C compiler gcc is popular and available for many platforms. PC users may also be familiar with the Borland bcc compiler.
There are also equivalent C++ compilers which are usually denoted by CC (note upper case CC. For example Sun provides CC and GNU GCC. The GNU compiler is also denoted by g++
Other (less common) C/C++ compilers exist. All the above compilers operate in essentially the same manner and share many common command line options. Below and in Appendix [*] we list and give example uses many of the common compiler options. However, the best source of each compiler is through the online manual pages of your system: e.g. man cc.
For the sake of compactness in the basic discussions of compiler operation we will simply refer to the cc compiler -- other compilers can simply be substituted in place of cc unless otherwise stated.
To Compile your program simply invoke the command cc. The command must be followed by the name of the (C) program you wish to compile. A number of compiler options can be specified also. We will not concern ourselves with many of these options yet, some useful and often essential options are introduced below.
Thus, the basic compilation command is:
cc program.c
where program.c is the name of the file.
If there are obvious errors in your program (such as mistypings, misspelling one of the key words or omitting a semi-colon), the compiler will detect and report them.
There may, of course, still be logical errors that the compiler cannot detect. You may be telling the computer to do the wrong operations.
When the compiler has successfully digested your program, the compiled version, or executable, is left in a file called a.out or if the compiler option -o is used : the file listed after the -o.
It is more convenient to use a -o and filename in the compilation as in
cc -o program program.c
which puts the compiled program into the file program (or any file you name following the "-o" argument) instead of putting it in the file a.out .
Running the program
The next stage is to actually run your executable program. To run an executable in UNIX, you simply type the name of the file containing it, in this case program (or a.out)
This executes your program, printing any results to the screen. At this stage there may be run-time errors, such as division by zero, or it may become evident that the program has produced incorrect output.
If so, you must return to edit your program source, and recompile it, and run it again.
The C Compilation Model
We will briefly highlight key features of the C Compilation model (Fig. 2.1) here.
The Preprocessor
We will study this part of the compilation process in greater detail later. However we need some basic information for some C programs.The Preprocessor accepts source code as input and is responsible for
- removing comments
- interpreting special preprocessor directives denoted by #.
- #include -- includes contents of a named file. Files usually called header files. e.g
- #include <math.h> -- standard library maths file.
- #include <stdio.h> -- standard library I/O file
- #define -- defines a symbolic name or constant. Macro substitution.
- #define MAX_ARRAY_SIZE 100
Tuesday, August 31, 2010
Week 7 Phase Test Week
Robotic interception of moving objects (continued from last week)
Among the most efficient approaches is APPE (Active Prediction, Planning and Execution) system for robotic
interception of moving objects. The key feature of the system is the ability of a robot to perform a task autonomously without complete a priori information.
Kalman filtering has proved very useful in autonomous and assisted navigation and guidance systems, radar tracking of moving objects, etc. The Kalman filter is a set of mathematical equations that provides an efficient computational (recursive) solution of the least-squares method. The filter is very powerful in several aspects: it supports estimates of past, present, and even future states, and it can do so even when the precise nature of the modelled system is unknown. Kalman filtering is also a computationally efficient algorithm, which generates an optimal estimate from a sequence of noisy observations.
This paper discuses an implementation of a robotic interception (a shoot function in robot soccer) based on image capture/processing combined with the successful use of Kalman filtering aiming at substantial improvement in shooting accuracy both for a stationary and moving object (the ball).
That's all for this week, Phase Test is in the way but from what written here i had understand that the article state that it use Kalman filtering to solve shooting accuracy problem when the ball is moving.
Among the most efficient approaches is APPE (Active Prediction, Planning and Execution) system for robotic
interception of moving objects. The key feature of the system is the ability of a robot to perform a task autonomously without complete a priori information.
Kalman filtering has proved very useful in autonomous and assisted navigation and guidance systems, radar tracking of moving objects, etc. The Kalman filter is a set of mathematical equations that provides an efficient computational (recursive) solution of the least-squares method. The filter is very powerful in several aspects: it supports estimates of past, present, and even future states, and it can do so even when the precise nature of the modelled system is unknown. Kalman filtering is also a computationally efficient algorithm, which generates an optimal estimate from a sequence of noisy observations.
This paper discuses an implementation of a robotic interception (a shoot function in robot soccer) based on image capture/processing combined with the successful use of Kalman filtering aiming at substantial improvement in shooting accuracy both for a stationary and moving object (the ball).
That's all for this week, Phase Test is in the way but from what written here i had understand that the article state that it use Kalman filtering to solve shooting accuracy problem when the ball is moving.
Tuesday, August 24, 2010
Week 6 control algorithm for the interception of a mobile target
This week i will study on the control algorithm to try understand more, so i had do some Google search and found this article;
In this article it say that, because of the dynamics and high complexity of the robot soccer system as well as maneuverability and speed of its robots, the accurate path planning and prediction of moving targets have gained special importance. Several techniques have been proposed for path planning including the use of genetic algorithms and fuzzy logic.
http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.78.2284&rep=rep1&type=pdf
In this article it say that, because of the dynamics and high complexity of the robot soccer system as well as maneuverability and speed of its robots, the accurate path planning and prediction of moving targets have gained special importance. Several techniques have been proposed for path planning including the use of genetic algorithms and fuzzy logic.
Coupled to the path-planning problem is obstacle avoidance in a dynamic environment. The path plans must be dynamically updated to reflect the changes in the environment, which means that they have to be created in real-time.
Robots have to exhibit basic actions like positioning at a designated location, moving to the ball or blocking an opponent, turning to a desired angle, circling around the ball and shooting the ball into opponent’s goal. Among other factors, the strategy and path planning in a robot soccer game are dependent on the ball position.
The robot's main sensor system is vision, which captures and processes the image at 30 to 60 frames per second. The odd and even scan fields of the interlaced image are processed separately. For this reason a stationary object like the ball is reported at different locations in each frame due to different quantization errors. The errors are compounded by the variation of light intensity from one frame to another. These quantization errors are inherent in the system and have a significant impact on the shooting accuracy even when the ball is stationary. However, the tests carried out on a moving ball are more significant because the interception accuracy suffers when the ball is moving. This is due to the fact that control actions are initiated based on the current ‘static’ state of objects whereas action must be taken based on predicted future positions.
That's all i had read for this week because i need to process all of this information to understand it.
Subscribe to:
Posts (Atom)